Поделиться через


Улучшения соответствия C++, изменения в поведении и исправления ошибок в Visual Studio 2017

В каждом выпуске Microsoft C/C++ в Visual Studio (MSVC) добавляются улучшения соответствия и исправления ошибок. В этой статье перечислены улучшения в разбивке по основным выпускам и версиям системы. Чтобы перейти непосредственно к изменениям в конкретной версии, используйте список, приведенный ниже в этой статье.

В этом документе перечислены изменения в Visual Studio 2017. Инструкции по изменениям в Visual Studio 2022 см. в статье Улучшения соответствия C++ в Visual Studio 2022. Инструкции по изменениям в Visual Studio 2019 см. в статье Улучшения соответствия C++ в Visual Studio 2019. Полный список предыдущих улучшений соответствия см. в статье Новые возможности Visual C++ 2003–2015.

Улучшения соответствия в Visual Studio 2017 RTW (версия 15.0)

Благодаря поддержке обобщенного выражения constexpr и NSDMI (нестатическая инициализация элементов данных) для статистических выражений, компилятор MSVC в Visual Studio 2017 теперь включает все функции, добавленные в стандарте C++14. Обратите внимание, что в компиляторе по-прежнему отсутствуют несколько функций из стандартов C++11 и C++98. Сведения о текущем состоянии компилятора см. в статье Соответствие стандартам языка Microsoft C/C++.

C++11: поддержка SFINAE выражения в дополнительных библиотеках

Продолжается улучшение поддержки выражения SFINAE в компиляторе. Это требуется для дедукции и подстановки аргументов шаблона, когда выражения decltype и constexpr могут отображаться в качестве параметров шаблона. Дополнительные сведения см. в разделе Усовершенствования выражения SFINAE в версии-кандидате Visual Studio 2017.

C++14: NSDMI для агрегатов

Агрегат — это массив или класс без предоставленного пользователем конструктора, без частных или защищенных нестатических элементов данных, без базовых классов и виртуальных функций. Начиная с версии C++14, агрегаты могут содержать инициализаторы элементов. Дополнительные сведения см. в разделе Инициализаторы членов и статистические выражения.

C++14: расширенный constexpr

Выражения, объявленные как constexpr, теперь могут содержать определенные виды объявлений, операторы if и switch, операторы циклов, а также изменения объектов, время существования которых началось с вычисления выражения constexpr. Кроме того, теперь не требуется, чтобы нестатическая функция-член constexpr явным образом была const. Дополнительные сведения см. в статье Нестрогие ограничения для функций constexpr.

C++17: Terse static_assert

Параметр сообщения для static_assert является необязательным. Дополнительные сведения см. в разделе N3928. Расширение static_assert версии 2.

C++17: атрибут [[fallthrough]]

В режиме /std:c++17 и более поздних атрибут [[fallthrough]] можно использовать в контексте операторов switch как указание для компилятора передать управление вниз. Этот атрибут позволит избежать вывода предупреждений компилятора в таких ситуациях. Дополнительные сведения см. в разделе P0188R0 - Wording for [[fallthrough]] attribute.

Обобщенный цикл for на основе диапазона

Циклы for на основе диапазона больше не требуют, чтобы begin() и end() возвращали объекты одного типа. Это изменение позволяет end() возвращать граничную метку согласно использованию в диапазонах range-v3 и технической спецификации по диапазонам (которая уже готова, но еще не опубликована). Дополнительные сведения см. в разделе P0184R0 - Generalizing the Range-Based for Loop.

Инициализация копии списка

Visual Studio 2017 правильно вызывает ошибки компилятора, связанные с созданием объектов с помощью списков инициализатора. Эти ошибки не перехватывались в Visual Studio 2015 и могли привести к сбоям или неопределенному поведению среды выполнения. В соответствии с N4594 13.3.1.7p1данными copy-list-initializationкомпилятор должен рассмотреть явный конструктор для разрешения перегрузки. Но он должен выдавать ошибку, если выбирается конкретно эта перегрузка.

Следующие два примера кода компилируются в Visual Studio 2015, но не в Visual Studio 2017.

struct A
{
    explicit A(int) {}
    A(double) {}
};

int main()
{
    A a1 = { 1 }; // error C3445: copy-list-initialization of 'A' cannot use an explicit constructor
    const A& a2 = { 1 }; // error C2440: 'initializing': cannot convert from 'int' to 'const A &'

}

Чтобы исправить эту ошибку, следует использовать прямую инициализацию:

A a1{ 1 };
const A& a2{ 1 };

В Visual Studio 2015 компилятор ошибочно интерпретировал инициализацию копии списка так же, как обычную инициализацию копии. Он рассматривал только преобразование конструкторов для разрешения перегрузки. В следующем примере Visual Studio 2015 выбирает MyInt(23). Visual Studio 2017 правильно выводит ошибку.

// From http://www.open-std.org/jtc1/sc22/wg21/docs/cwg_closed.html#1228
struct MyStore {
    explicit MyStore(int initialCapacity);
};

struct MyInt {
    MyInt(int i);
};

struct Printer {
    void operator()(MyStore const& s);
    void operator()(MyInt const& i);
};

void f() {
    Printer p;
    p({ 23 }); // C3066: there are multiple ways that an object
        // of this type can be called with these arguments
}

Этот пример аналогичен предыдущему, но выводит другую ошибку. Он выполняется успешно в Visual Studio 2015, а в Visual Studio 2017 завершается ошибкой C2668.

struct A {
    explicit A(int) {}
};

struct B {
    B(int) {}
};

void f(const A&) {}
void f(const B&) {}

int main()
{
    f({ 1 }); // error C2668: 'f': ambiguous call to overloaded function
}

Нерекомендуемые определения типов

Теперь Visual Studio 2017 выдает правильное предупреждение для нерекомендуемых определений типов, объявленных в классе или структуре. В следующем примере показана компиляция без предупреждений в Visual Studio 2015. Но в Visual Studio 2017 выводится ошибка C4996.

struct A
{
    // also for __declspec(deprecated)
    [[deprecated]] typedef int inttype;
};

int main()
{
    A::inttype a = 0; // C4996 'A::inttype': was declared deprecated
}

constexpr

Visual Studio 2017 правильно выдает ошибку, если левый операнд в операции условного вычисления является недопустимым в контексте constexpr. Следующий код компилируется в Visual Studio 2015, но не в Visual Studio 2017, где он вызывает ошибку C3615:

template<int N>
struct array
{
    int size() const { return N; }
};

constexpr bool f(const array<1> &arr)
{
    return arr.size() == 10 || arr.size() == 11; // C3615 constexpr function 'f' cannot result in a constant expression
}

Чтобы исправить эту ошибку, объявите функцию array::size() как constexpr или удалите квалификатор constexpr из f.

Типы классов, передаваемые в функции с переменным числом аргументов

В Visual Studio 2017 классы и структуры, передаваемые в вариадическую функцию, например printf, должны быть доступны для простого копирования. При передаче таких объектов компилятор просто выполняет побитовое копирование и не вызывает конструктор или деструктор.

#include <atomic>
#include <memory>
#include <stdio.h>

int main()
{
    std::atomic<int> i(0);
    printf("%i\n", i); // error C4839: non-standard use of class 'std::atomic<int>'
                        // as an argument to a variadic function.
                        // note: the constructor and destructor will not be called;
                        // a bitwise copy of the class will be passed as the argument
                        // error C2280: 'std::atomic<int>::atomic(const std::atomic<int> &)':
                        // attempting to reference a deleted function

    struct S {
        S(int i) : i(i) {}
        S(const S& other) : i(other.i) {}
        operator int() { return i; }
    private:
        int i;
    } s(0);
    printf("%i\n", s); // warning C4840 : non-portable use of class 'main::S'
                      // as an argument to a variadic function
}

Чтобы исправить эту ошибку, можно вызвать функцию-член, возвращающую тип, доступный для простого копирования,

    std::atomic<int> i(0);
    printf("%i\n", i.load());

или выполнить статическое приведение для преобразования объекта перед его передачей:

    struct S {/* as before */} s(0);
    printf("%i\n", static_cast<int>(s))

Для строк, созданных и управляемых с помощью CString, следует использовать предоставленный operator LPCTSTR() для приведения объекта CString к указателю C, ожидаемому строкой формата.

CString str1;
CString str2 = _T("hello!");
str1.Format(_T("%s"), static_cast<LPCTSTR>(str2));

CV-квалификаторы в конструкторах класса

В Visual Studio 2015 компилятор иногда ошибочно игнорирует cv-квалификатор при создании объекта класса путем вызова конструктора. Эта проблема может привести к сбою или непредсказуемому поведению среды выполнения. Следующий пример компилируется в Visual Studio 2015, но вызывает ошибку компилятора в Visual Studio 2017:

struct S
{
    S(int);
    operator int();
};

int i = (const S)0; // error C2440

Чтобы исправить эту ошибку, объявите operator int() как const.

Проверка доступа к полным именам в шаблонах

В предыдущих версиях компилятора не выполнялась проверка доступа к полным именам в некоторых контекстах шаблонов. Эта проблема могла помешать ожидаемой работе SFINAE там, где подстановка должна завершиться ошибкой из-за отсутствия доступа к имени. Это могло приводить к сбою или неожиданному поведению среды выполнения из-за того, что компилятор ошибочно вызывал неверную перегрузку оператора. В Visual Studio 2017 выводится ошибка компилятора. Ошибки могут различаться, но, как правило, это ошибка C2672, "Не найдена соответствующая перегруженная функция". Следующий код компилируется в Visual Studio 2015, но вызывает ошибку в Visual Studio 2017:

#include <type_traits>

template <class T> class S {
    typedef typename T type;
};

template <class T, std::enable_if<std::is_integral<typename S<T>::type>::value, T> * = 0>
bool f(T x);

int main()
{
    f(10); // C2672: No matching overloaded function found.
}

Отсутствующие списки аргументов шаблона

В Visual Studio 2015 и более ранних версиях компилятор не обнаруживал все отсутствующие списки аргументов шаблона. Он не заметил бы, если бы отсутствующий шаблон отобразился в списке параметров шаблона (например, когда аргумент шаблона по умолчанию или параметр шаблона, не являющийся типом, отсутствовали). Эта проблема может привести к непредсказуемому поведению, включая сбои компилятора или непредсказуемое поведение среды выполнения. Следующий код компилируется в Visual Studio 2015, но выводит ошибку в Visual Studio 2017.

template <class T> class ListNode;
template <class T> using ListNodeMember = ListNode<T> T::*;
template <class T, ListNodeMember M> class ListHead; // C2955: 'ListNodeMember': use of alias
                                                     // template requires template argument list

// correct:  template <class T, ListNodeMember<T> M> class ListHead;

Выражение SFINAE

Чтобы реализовать поддержку выражения SFINAE, компилятор теперь анализирует аргументы decltype при объявлении шаблонов, а не при создании их экземпляров. Поэтому независимая специализация, обнаруженная в аргументе decltype, не откладывается до момента создания экземпляра. Она обрабатывается немедленно и в этот же момент проверяется на все возможные ошибки.

В следующем примере показана такая ошибка компилятора, возникающая во время объявления.

#include <utility>
template <class T, class ReturnT, class... ArgsT>
class IsCallable
{
public:
    struct BadType {};

    template <class U>
    static decltype(std::declval<T>()(std::declval<ArgsT>()...)) Test(int); //C2064. Should be declval<U>

    template <class U>
    static BadType Test(...);

    static constexpr bool value = std::is_convertible<decltype(Test<T>(0)), ReturnT>::value;
};

constexpr bool test1 = IsCallable<int(), int>::value;
static_assert(test1, "PASS1");
constexpr bool test2 = !IsCallable<int*, int>::value;
static_assert(test2, "PASS2");

Классы, объявляемые в анонимных пространствах имен

Согласно стандарту C++, объявленный внутри анонимного пространства имен класс имеет внутреннюю компоновку и не может быть экспортирован. В Visual Studio 2015 и более ранних версиях это правило не применялось. В Visual Studio 2017 это правило применяется частично. В Visual Studio 2017 в следующем примере возникает ошибка C2201:

struct __declspec(dllexport) S1 { virtual void f() {} };
  // C2201 const anonymous namespace::S1::vftable: must have external linkage
  // in order to be exported/imported.

Инициализаторы по умолчанию для членов класса значения (C++/CLI)

В Visual Studio 2015 и более ранних версиях компилятор допускал (но игнорировал) инициализатор членов по умолчанию для члена класса значений. Инициализатор по умолчанию для класса значений всегда инициализирует члены нулевым значением. Конструктор по умолчанию не допускается. В Visual Studio 2017 инициализаторы членов по умолчанию вызывают ошибку компилятора, как показано в следующем примере:

value struct V
{
    int i = 0; // error C3446: 'V::i': a default member initializer
               // isn't allowed for a member of a value class
};

Индексаторы по умолчанию (C++/CLI)

В Visual Studio 2015 и более ранних версиях в некоторых случаях компилятор неправильно определял свойство по умолчанию как индексатор по умолчанию. Эту проблему удавалось решить благодаря использованию идентификатора default для доступа к свойству. Это обходное решение само стало создавать проблемы после того, как значение default было введено как ключевое слово в C++11. В Visual Studio 2017 исправлены ошибки, которые требовали обходного решения. Теперь компилятор выдает ошибку, если для доступа к свойству по умолчанию для класса используется default.

//class1.cs

using System.Reflection;
using System.Runtime.InteropServices;

namespace ClassLibrary1
{
    [DefaultMember("Value")]
    public class Class1
    {
        public int Value
        {
            // using attribute on the return type triggers the compiler bug
            [return: MarshalAs(UnmanagedType.I4)]
            get;
        }
    }
    [DefaultMember("Value")]
    public class Class2
    {
        public int Value
        {
            get;
        }
    }
}

// code.cpp
#using "class1.dll"

void f(ClassLibrary1::Class1 ^r1, ClassLibrary1::Class2 ^r2)
{
       r1->Value; // error
       r1->default;
       r2->Value;
       r2->default; // error
}

В Visual Studio 2017 оба свойства значения доступны по их именам:

#using "class1.dll"

void f(ClassLibrary1::Class1 ^r1, ClassLibrary1::Class2 ^r2)
{
       r1->Value;
       r2->Value;
}

Улучшения соответствия в версии 15.3

Лямбда-выражения constexpr

Лямбда-выражения теперь можно использоваться в константных выражениях. Дополнительные сведения см. в разделе Лямбда-выражения constexpr в C++.

if constexpr в шаблонах функций

Шаблон функции может содержать операторы if constexpr для поддержки ветвления во время компиляции. Дополнительные сведения см. в статье Инструкции if constexpr.

Операторы выбора с инициализаторами

Оператор if может включать инициализатор, который вводит переменную в области видимости блока внутри самого оператора. Дополнительные сведения см. в разделе Операторы if с инициализатором.

Атрибуты [[maybe_unused]] и [[nodiscard]]

Новый атрибут [[maybe_unused]] подавляет предупреждения о неиспользовании сущности. Атрибут [[nodiscard]] создает предупреждение, если при вызове функции отбрасывается возвращаемое значение. Дополнительные сведения см. в статье Attributes in C++ (Атрибуты в C++).

Использование пространств имен атрибутов без повторения

Новый синтаксис позволяет включать в список атрибутов всего один идентификатор пространства имен. Дополнительные сведения см. в статье Attributes in C++ (Атрибуты в C++).

Структурированные привязки

Теперь можно одним объявлением сохранить значение с отдельными именами компонентов, когда значение является массивом, std::tuple или std::pair, либо когда все его нестатические элементы данных являются открытыми. Дополнительные сведения см. в разделе P0144R0 - Structured Bindings и возврате нескольких значений из функции.

Правила создания для значений enum class

Теперь происходит неявное преобразование для ограниченных перечислений, которое является не ограничивающим. Преобразование выполняется из базового типа ограниченного перечисления в само перечисление. Преобразование доступно, если его определение не вводит перечислитель и если источник использует синтаксис инициализации списка. Дополнительные сведения: P0138R2 - Construction Rules for enum class Values и Перечисления.

Захват *this по значению

Теперь лямбда-выражение может обращаться к объекту *this по значению. Это позволяет использовать лямбда-выражения в сценариях с параллельными и асинхронными операциями, особенно на новых архитектурах компьютеров. Дополнительные сведения см. в разделе P0018R3 - Lambda Capture of *this by Value as [=,*this].

Удаление operator++ для bool

operator++ больше не поддерживается для типов bool. Дополнительные сведения см. в разделе P0002R1 - Remove Deprecated operator++(bool).

Удаление нерекомендуемого ключевого слова register

Ключевое слово register, ранее признанное нерекомендуемым (и игнорируемое компилятором), теперь удалено из языка. Дополнительные сведения см. в разделе P0001R1 - Remove Deprecated Use of the register Keyword.

Вызовы шаблонов удаленного элемента

В предыдущих версиях Visual Studio компилятор в некоторых случаях не сообщал об ошибках при некорректных вызовах шаблона удаленного элемента. Эти вызовы могли вызвать сбои во время выполнения. Следующий код теперь возвращает ошибку C2280:

template<typename T>
struct S {
   template<typename U> static int f() = delete;
};

void g()
{
   decltype(S<int>::f<int>()) i; // this should fail with
// C2280: 'int S<int>::f<int>(void)': attempting to reference a deleted function
}

Чтобы устранить эту ошибку, объявите элемент i как int.

Проверки предварительных условий для признаков типов

В Visual Studio 2017 версии 15.3 улучшены проверки предварительных условий для признаков типов на более строгое следование стандарту. Одна из проверок проверяет присвоение. В Visual Studio 2017 версии 15.3 следующий код возвращает ошибку C2139:

struct S;
enum E;

static_assert(!__is_assignable(S, S), "fail"); // C2139 in 15.3
static_assert(__is_convertible_to(E, E), "fail"); // C2139 in 15.3

Новое предупреждение компилятора и проверки среды выполнения для маршалинга между собственными и управляемыми функциями

Вызов собственных функций из управляемых функций требует маршалинга. Среда CLR выполняет такой маршалинг, но не понимает семантику C++. Если вы передадите собственный объект по значению, CLR вызовет конструктор копии объекта или применит функцию BitBlt, что может привести к непредсказуемому поведению в среде выполнения.

Теперь компилятор выдает предупреждение, если во время компиляции находит ошибку, что через границу собственной и управляемой функции передается по значению собственный объект с удаленным конструктором копии. Если компилятор во время компиляции не может получить такую информацию, он внедряет проверку времени выполнения, чтобы программа немедленно вызвала std::terminate в случае некорректного маршалинга. В Visual Studio 2017 версии 15.3 следующий код вызывает предупреждение C4606:

class A
{
public:
   A() : p_(new int) {}
   ~A() { delete p_; }

   A(A const &) = delete;
   A(A &&rhs) {
   p_ = rhs.p_;
}

private:
   int *p_;
};

#pragma unmanaged

void f(A a)
{
}

#pragma managed

int main()
{
    // This call from managed to native requires marshaling. The CLR doesn't
    // understand C++ and uses BitBlt, which results in a double-free later.
    f(A()); // C4606 'A': passing argument by value across native and managed
    // boundary requires valid copy constructor. Otherwise, the runtime
    // behavior is undefined.`
}

Чтобы устранить такую ошибку, удалите директиву #pragma managed, чтобы вызывающая функция стала собственной. Это позволит избежать маршалинга.

Экспериментальные предупреждения API для WinRT

API-интерфейсы WinRT, выпускаемые для экспериментов и обратной связи, обозначаются атрибутом Windows.Foundation.Metadata.ExperimentalAttribute. В Visual Studio 2017 версии 15.3 при обнаружении этого атрибута компилятор выдает предупреждение C4698. Некоторые API-интерфейсы в предыдущих версиях Windows SDK уже помечены этим атрибутом, поэтому обращения к этим API теперь вызывают указанное выше предупреждение. В новых пакетах SDK для Windows атрибут удален из всех поставляемых типов. Если вы используете более раннюю версию пакета SDK, необходимо подавить эти предупреждения для всех вызовов к поставляемым типам.

Следующий код теперь вызывает предупреждение C4698:

Windows::Storage::IApplicationDataStatics2::GetForUserAsync(); // C4698
// 'Windows::Storage::IApplicationDataStatics2::GetForUserAsync' is for
// evaluation purposes only and is subject to change or removal in future updates

Чтобы отключить это предупреждение, добавьте атрибут #pragma.

#pragma warning(push)
#pragma warning(disable:4698)

Windows::Storage::IApplicationDataStatics2::GetForUserAsync();

#pragma warning(pop)

Определение функции-члена шаблона вне строки

В Visual Studio 2017 версии 15.3 появляется сообщение об ошибке, если обнаруживается внешнее определение компонентной функции шаблона, не объявленное в классе. Следующий код теперь возвращает ошибку C2039:

struct S {};

template <typename T>
void S::f(T t) {} // C2039: 'f': is not a member of 'S'

Чтобы исправить такую ошибку, добавьте в класс следующее объявление:

struct S {
    template <typename T>
    void f(T t);
};
template <typename T>
void S::f(T t) {}

Попытка получить адрес для указателя this

В C++ this является значением prvalue с типом указателя для X. Вы не можете получить адрес this или привязать его к ссылке на lvalue. В предыдущих версиях Visual Studio компилятор позволял обойти это ограничение, выполнив приведение типов. В Visual Studio 2017 версии 15.3 компилятор выдает ошибку C2664.

Преобразование в недоступный базовый класс

Visual Studio 2017 версии 15.3 выводит сообщение об ошибке при попытке преобразовать тип в недоступный базовый класс. Следующий код является некорректным и может привести к сбою в среде выполнения. Теперь компилятор создает ошибку C2243 для такого кода:

#include <memory>

class B { };
class D : B { }; // C2243: 'type cast': conversion from 'D *' to 'B *' exists, but is inaccessible

void f()
{
   std::unique_ptr<B>(new D());
}

Аргументы по умолчанию не допускаются во внешних определениях функций-членов

Аргументы по умолчанию не допускаются во внешних определениях функций-членов в классах шаблонов. Компилятор выдаст предупреждение в режиме /permissive и критическую ошибку в режиме /permissive-.

В предыдущих версиях Visual Studio следующий некорректный код может вызвать сбой среды выполнения. В Visual Studio 2017 версии 15.3 возникает предупреждение C5037:

template <typename T>
struct A {
    T f(T t, bool b = false);
};

template <typename T>
T A<T>::f(T t, bool b = false) // C5037: 'A<T>::f': an out-of-line definition of a member of a class template cannot have default arguments
{
    // ...
}

Чтобы устранить такую ошибку, удалите аргумент по умолчанию = false.

Использование offsetof с обозначением составного члена

Если используется offsetof(T, m), где m является "обозначением составного элемента", в Visual Studio 2017 версии 15.3 появится предупреждение при компиляции с параметром /Wall. Следующий код является некорректным и может привести к сбою во время выполнения. В Visual Studio 2017 версии 15.3 возникает предупреждение C4841:

struct A {
   int arr[10];
};

// warning C4841: non-standard extension used: compound member designator used in offsetof
constexpr auto off = offsetof(A, arr[2]);

Чтобы исправить этот код, отключите предупреждение с помощью директивы pragma или уберите из кода offsetof.

#pragma warning(push)
#pragma warning(disable: 4841)
constexpr auto off = offsetof(A, arr[2]);
#pragma warning(pop)

Использование offsetof со статическим элементом данных или функцией-членом

Если используется offsetof(T, m), где m ссылается на статические данные-член или на функцию-член, Visual Studio 2017 версии 15.3 выдает ошибку. Следующий код возвращает ошибку C4597:

#include <cstddef>

struct A {
   int ten() { return 10; }
   static constexpr int two = 2;
};

constexpr auto off = offsetof(A, ten);  // C4597: undefined behavior: offsetof applied to member function 'A::ten'
constexpr auto off2 = offsetof(A, two); // C4597: undefined behavior: offsetof applied to static data member 'A::two'

Этот код является некорректным и может привести к сбою во время выполнения. Чтобы устранить такую ошибку, измените код так, чтобы он не создавал неопределенное поведение. Это код не является переносимым и запрещен стандартом C++.

Новое предупреждение для __declspec атрибутов

В Visual Studio 2017 версии 15.3 компилятор больше не игнорирует атрибуты, если __declspec(...) применяется перед спецификацией компоновки extern "C". В прошлом компилятор игнорировал такой атрибут, что могло повлиять на работу в среде выполнения. Если заданы параметры /Wall и /WX, следующий код вызывает предупреждение C4768:

__declspec(noinline) extern "C" HRESULT __stdcall // C4768: __declspec attributes before linkage specification are ignored

Чтобы устранить это предупреждение, переместите extern "C" вперед:

extern "C" __declspec(noinline) HRESULT __stdcall

Это предупреждение по умолчанию отключено в Visual Studio 2017 версии 15.3 и влияет только на скомпилированный /Wall /WXкод. Начиная с Visual Studio 2017 версии 15.5 предупреждение по умолчанию включено как предупреждение уровня 3.

decltype и вызовы удаленных деструкторов

В предыдущих версиях Visual Studio компилятор не отслеживал возникновение вызовов удаленных деструкторов в контексте выражений, связанных с decltype. В Visual Studio 2017 версии 15.3 следующий код возвращает ошибку C2280:

template<typename T>
struct A
{
   ~A() = delete;
};

template<typename T>
auto f() -> A<T>;

template<typename T>
auto g(T) -> decltype((f<T>()));

void h()
{
   g(42); // C2280: 'A<T>::~A(void)': attempting to reference a deleted function
}

Неинициализированные переменные const

В выпуске Visual Studio 2017 RTW использовалась регрессия, из-за которой компилятор C++ не выдавал диагностических сообщений о том, что переменная const не инициализирована. Эта регрессия была устранена в Visual Studio 2017 версии 15.3. Следующий код теперь вызывает предупреждение C4132:

const int Value; // C4132: 'Value': const object should be initialized

Чтобы исправить эту ошибку, присвойте значение переменной Value.

Пустые объявления

Теперь Visual Studio 2017 версии 15.3 предупреждает о пустых объявлениях для всех типов, а не только для встроенных. Теперь для следующего кода создаются предупреждения C4091 уровня 2 для всех четырех объявлений:

struct A {};
template <typename> struct B {};
enum C { c1, c2, c3 };

int;    // warning C4091 : '' : ignored on left of 'int' when no variable is declared
A;      // warning C4091 : '' : ignored on left of 'main::A' when no variable is declared
B<int>; // warning C4091 : '' : ignored on left of 'B<int>' when no variable is declared
C;      // warning C4091 : '' : ignored on left of 'C' when no variable is declared

Чтобы устранить эти предупреждения, закомментируйте или удалите все пустые объявления. В случаях, когда объекты без имен созданы специально для побочных эффектов (например, RAII), им следует присвоить имена.

Это предупреждение отключено при использовании /Wv:18 и включено по умолчанию на уровне предупреждений W2.

std::is_convertible для типов массивов

В предыдущих версиях компилятора класс std::is_convertible выдавал неверные результаты для типов массива. Из-за этого разработчикам библиотек необходимо было особым образом обрабатывать в компиляторе Microsoft C++ признаки типа std::is_convertible<...>. В следующем примере статические функции assert успешно срабатывают в предыдущих версиях Visual Studio, но в Visual Studio 2017 версии 15.3 завершаются ошибкой:

#include <type_traits>

using Array = char[1];

static_assert(std::is_convertible<Array, Array>::value);
static_assert(std::is_convertible<const Array, const Array>::value, "");
static_assert(std::is_convertible<Array&, Array>::value, "");
static_assert(std::is_convertible<Array, Array&>::value, "");

std::is_convertible<From, To> проверяет, правильно ли сформировано определение мнимой функции.

   To test() { return std::declval<From>(); }

Частные деструкторы и std::is_constructible

Предыдущие версии компилятора не проверяли, является ли деструктор закрытым, при получении результата std::is_constructible. Теперь это принимается во внимание. В следующем примере статические функции assert успешно срабатывают в предыдущих версиях Visual Studio, но в Visual Studio 2017 версии 15.3 завершаются ошибкой:

#include <type_traits>

class PrivateDtor {
   PrivateDtor(int) { }
private:
   ~PrivateDtor() { }
};

// This assertion used to succeed. It now correctly fails.
static_assert(std::is_constructible<PrivateDtor, int>::value);

Закрытые деструкторы могут сделать тип неконструируемым. При вычислении std::is_constructible<T, Args...> подразумевается следующее объявление.

   T obj(std::declval<Args>()...)

Этот вызов подразумевает вызов деструктора.

C2668: неоднозначное разрешение перегрузки

Предыдущим версиям компилятора иногда не удавалось обнаружить неоднозначность, если при использовании объявлений и при поиске функций по аргументам выявлялись различные кандидаты. Эта ошибка может привести к некорректному выбору перегрузки и непредсказуемому поведению в среде выполнения. В следующем примере Visual Studio 2017 версии 15.3 правильно выдает ошибку C2668:

namespace N {
   template<class T>
   void f(T&, T&);

   template<class T>
   void f();
}

template<class T>
void f(T&, T&);

struct S {};
void f()
{
   using N::f;

   S s1, s2;
   f(s1, s2); // C2668: 'f': ambiguous call to overloaded function
}

Чтобы исправить код, удалите оператор using N::f, если вы намереваетесь вызвать ::f().

C2660: локальные объявления функций и проверка по аргументам

При локальном объявлении функции скрываются внутри области, поэтому поиск по аргументам не осуществляется. В этом случае предыдущие версии компилятора всегда выполняли поиск с учетом аргументов. Это потенциально может привести к непредвиденному поведению во время выполнения, если компилятор выберет неправильную перегрузку. Как правило, такая ошибка связана с ошибкой в сигнатуре локального объявления функции. В следующем примере Visual Studio 2017 версии 15.3 правильно выдает ошибку C2660:

struct S {};
void f(S, int);

void g()
{
   void f(S); // C2660 'f': function does not take 2 arguments:
   // or void f(S, int);
   S s;
   f(s, 0);
}

Чтобы устранить эту проблему, измените сигнатуру f(S) или удалите ее.

C5038: порядок инициализации в списках инициализатора

Элементы класса инициализируются в порядке их объявления, а не в порядке отображения в списках инициализаторов. В предыдущих версиях компилятора, когда порядок списка инициализаторов отличался от порядка объявления, никакого предупреждения не выводилось. Эта проблема могла приводить к неопределенному поведению среды выполнения, если инициализация одного элемента зависела от другого элемента, инициализированного в списке. В следующем примере Visual Studio 2017 версии 15.3 (с /Wall) выдает предупреждение C5038:

struct A
{    // Initialized in reverse, y reused
    A(int a) : y(a), x(y) {} // C5038: data member 'A::y' will be initialized after data member 'A::x'
    int x;
    int y;
};

Чтобы устранить проблему, список инициализатора должен иметь тот же порядок, что и объявления. Похожее предупреждение возникает, когда один инициализатор или оба ссылаются на члены базового класса.

Это предупреждение по умолчанию отключено и относится только к коду, скомпилированному с параметром /Wall.

Улучшения соответствия в версии 15.5

Функции, отмеченные как [14], доступны без дополнительных условий даже в режиме /std:c++14.

Новый параметр компилятора для extern constexpr

В более ранних версиях Visual Studio компилятор всегда использовал для переменной constexpr внутреннюю компоновку, даже если эта переменная была помечена как extern. В Visual Studio 2017 версии 15.5 новый параметр компилятора (/Zc:externConstexpr) обеспечивает корректное поведение, соответствующее стандартам. Дополнительные сведения см. в разделе Связывание extern constexpr.

Удаление динамических спецификаций исключений

P0003R5: динамические спецификации исключений в C++11 стали нерекомендуемыми. Эта функция удалена из C++17, но (по-прежнему) нерекомендуемая спецификация throw() сохраняется исключительно в качестве псевдонима для noexcept(true). Дополнительные сведения см. в разделе Удаление спецификации динамических исключений и noexcept.

not_fn()

P0005R4: not_fn является заменой not1 и not2.

Изменение формулировки для enable_shared_from_this

P0033R1: класс enable_shared_from_this добавлен в C++11. Стандарт C++17 обновляет спецификацию, чтобы лучше обрабатывать некоторые пограничные случаи. [14]

Соединение карт и наборов

P0083R3. Эта возможность позволяет извлекать узлы из ассоциативных контейнеров (например, map, set, unordered_map, unordered_set), а затем изменять их и вставлять обратно в тот же или другой контейнер с тем же типом узла. (Типичный вариант использования — извлечение узла из std::map, изменение ключа и повторная вставка.)

Нерекомендуемые части библиотек

P0174R2. Со временем несколько функций стандартной библиотеки C++ были заменены новыми, признаны нецелесообразными или проблемными. Эти функции являются официально нерекомендуемыми в C++17.

Удаление поддержки распределителя в std::function

P0302R1. До появления C++17 шаблон класса std::function содержал несколько конструкторов, принимающих аргумент распределителя. Однако использование распределителей в данном контексте было проблематичным, а семантика — неясной. Проблема конструкторов теперь устранена.

Исправления для not_fn()

P0358R1. Новая формулировка std::not_fn поддерживает передачу категории значения при вызове класса-оболочки.

shared_ptr<T[]>, shared_ptr<T[N]>

P0414R2: внесение изменений shared_ptr из Library Fundamentals в C++17. [14]

Исправление shared_ptr для массивов

P0497R0: исправления поддержки shared_ptr для массивов. [14]

Пояснение insert_return_type

P0508R0: ассоциативные и неупорядоченные контейнеры с уникальными ключами имеют функцию-член insert, возвращающую insert_return_type вложенного типа. Тип возвращаемого значения теперь определяется как специализация типа, который параметризуется по итератору и типу узла контейнера.

Встроенные переменные для стандартной библиотеки

Для P0607R0 несколько распространенных переменных, объявленных в стандартной библиотеке, теперь объявляются как встроенные.

Нерекомендуемые функции в приложении D

Приложение D стандарта C++ содержит все функции, признанные нерекомендуемыми, включая shared_ptr::unique(), <codecvt> и namespace std::tr1. Если задан параметр компилятора /std:c++17 или более поздней версии, почти все функции стандартной библиотеки в приложении D помечаются как нерекомендуемые. Дополнительные сведения см. в этой статье.

Пространство имен std::tr2::sys в <experimental/filesystem> теперь по умолчанию выдает предупреждение о нерекомендуемых функциях при использовании /std:c++14, а при использовании /std:c++17 и более поздних версий оно по умолчанию удалено.

Улучшено соответствие в <iostream> за счет отказа от нестандартного расширения (явных специализаций в классе).

Стандартная библиотека теперь использует шаблоны переменных внутренним образом.

Стандартная библиотека была обновлена в соответствии с изменениями компилятора C++17. В обновления входит добавление noexcept в систему типов, а также удаление спецификаций динамических исключений.

Частичное упорядочение изменений

Компилятор теперь правильно отклоняет следующий код и выводит нужное сообщение об ошибке:

template<typename... T>
int f(T* ...)
{
    return 1;
}

template<typename T>
int f(const T&)
{
    return 2;
}

int main()
{
    int i = 0;
    f(&i);    // C2668
}
t161.cpp
t161.cpp(16): error C2668: 'f': ambiguous call to overloaded function
t161.cpp(8): note: could be 'int f<int*>(const T &)'
        with
        [
            T=int*
        ]
t161.cpp(2): note: or       'int f<int>(int*)'
t161.cpp(16): note: while trying to match the argument list '(int*)'

Проблема в приведенном выше примере заключается в том, что имеется два различия в типах (const и non-const и pack и non-pack). Чтобы устранить ошибку компилятора, удалите одно из различий. Затем компилятор может однозначно упорядочить функции.

template<typename... T>
int f(T* ...)
{
    return 1;
}

template<typename T>
int f(T&)
{
    return 2;
}

int main()
{
    int i = 0;
    f(&i);
}

Обработчики исключений

Обработчики ссылок на тип массива или функции никогда не соответствуют никаким объектам исключений. Теперь компилятор верно учитывает это правило и выдает предупреждение C4843 уровня 4. Он также больше не соответствует обработчику char* или wchar_t* в строковом литерале, если используется /Zc:strictStrings.

int main()
{
    try {
        throw "";
    }
    catch (int (&)[1]) {} // C4843 (This should always be dead code.)
    catch (void (&)()) {} // C4843 (This should always be dead code.)
    catch (char*) {} // This should not be a match under /Zc:strictStrings
}
warning C4843: 'int (&)[1]': An exception handler of reference to array or function type is unreachable, use 'int*' instead
warning C4843: 'void (__cdecl &)(void)': An exception handler of reference to array or function type is unreachable, use 'void (__cdecl*)(void)' instead

Следующий код позволяет избежать этой ошибки:

catch (int (*)[1]) {}

Пространство имен std::tr1 не рекомендуется к использованию

Нестандартное пространство имен std::tr1 теперь помечается как нерекомендуемое в режимах C++14 и C++17. В Visual Studio 2017 версии 15.5 приведенный ниже код вызывает предупреждение C4996:

#include <functional>
#include <iostream>
using namespace std;

int main() {
    std::tr1::function<int (int, int)> f = std::plus<int>(); //C4996
    cout << f(3, 5) << std::endl;
    f = std::multiplies<int>();
    cout << f(3, 5) << std::endl;
}
warning C4996: 'std::tr1': warning STL4002: The non-standard std::tr1 namespace and TR1-only machinery are deprecated and will be REMOVED. You can define _SILENCE_TR1_NAMESPACE_DEPRECATION_WARNING to acknowledge that you have received this warning.

Чтобы исправить ошибку, удалите ссылку на пространство имен tr1.

#include <functional>
#include <iostream>
using namespace std;

int main() {
    std::function<int (int, int)> f = std::plus<int>();
    cout << f(3, 5) << std::endl;
    f = std::multiplies<int>();
    cout << f(3, 5) << std::endl;
}

Функции стандартной библиотеки в приложении D помечены как устаревшие

Если задан параметр режима компилятора /std:c++17 или более поздних версий, почти все функции стандартной библиотеки в приложении D помечаются как нерекомендуемые.

В Visual Studio 2017 версии 15.5 приведенный ниже код вызывает предупреждение C4996:

#include <iterator>

class MyIter : public std::iterator<std::random_access_iterator_tag, int> {
public:
    // ... other members ...
};

#include <type_traits>

static_assert(std::is_same<MyIter::pointer, int*>::value, "BOOM");
warning C4996: 'std::iterator<std::random_access_iterator_tag,int,ptrdiff_t,_Ty*,_Ty &>::pointer': warning STL4015: The std::iterator class template (used as a base class to provide typedefs) is deprecated in C++17. (The <iterator> header is NOT deprecated.) The C++ standard has never required user-defined iterators to derive from std::iterator. To fix this warning, stop deriving from std::iterator and start providing publicly accessible typedefs named iterator_category, value_type, difference_type, pointer, and reference. Note that value_type is required to be non-const, even for constant iterators. You can define _SILENCE_CXX17_ITERATOR_BASE_CLASS_DEPRECATION_WARNING or _SILENCE_ALL_CXX17_DEPRECATION_WARNINGS to acknowledge that you have received this warning.

Чтобы устранить ошибку, следуйте инструкциям в тексте предупреждения, как показано в следующем коде:

#include <iterator>

class MyIter {
public:
    typedef std::random_access_iterator_tag iterator_category;
    typedef int value_type;
    typedef ptrdiff_t difference_type;
    typedef int* pointer;
    typedef int& reference;

    // ... other members ...
};

#include <type_traits>

static_assert(std::is_same<MyIter::pointer, int*>::value, "BOOM");

Неиспользуемые локальные переменные

В Visual Studio версии 15.5 предупреждение C4189 создается в большинстве случаев, как показано в следующем коде:

void f() {
    char s[2] = {0}; // C4189. Either use the variable or remove it.
}
warning C4189: 's': local variable is initialized but not referenced

Чтобы устранить ошибку, удалите неиспользуемую переменную.

Однострочные комментарии

В Visual Studio 2017 версии 15.5 компилятор C больше не создает предупреждения C4001 и C4179. Ранее они создавались только при использовании параметра компилятора /Za. Эти предупреждения больше не нужны, так как однострочные комментарии входят в стандарт C, начиная с версии C99.

/* C only */
#pragma warning(disable:4001) // C4619
#pragma warning(disable:4179)
// single line comment
//* single line comment */
warning C4619: #pragma warning: there is no warning number '4001'

Если код не должен поддерживать обратную совместимость, отключите эти предупреждения, удалив подавление предупреждений C4001 и C4179. Если код должен поддерживать обратную совместимость, подавите вывод только предупреждения C4619.

/* C only */

#pragma warning(disable:4619)
#pragma warning(disable:4001)
#pragma warning(disable:4179)

// single line comment
/* single line comment */

Атрибуты __declspec с компоновкой extern "C"

В более ранних версиях Visual Studio компилятор игнорировал атрибуты __declspec(...), когда __declspec(...) был применен перед спецификацией компоновки extern "C". Это поведение приводило к созданию кода, который не планировался пользователем, с возможными последствиями для среды выполнения. Предупреждение C4768 было добавлено в Visual Studio версии 15.3, но было отключено по умолчанию. В Visual Studio 2017 версии 15.5 предупреждение включено по умолчанию.

__declspec(noinline) extern "C" HRESULT __stdcall // C4768
warning C4768: __declspec attributes before linkage specification are ignored

Чтобы устранить ошибку, поместите спецификацию компоновки перед атрибутом __declspec:

extern "C" __declspec(noinline) HRESULT __stdcall

Это новое предупреждение C4768 выдается для некоторых заголовков Windows SDK, которые поставлялись с Visual Studio 2017 15.3 или более ранними версиями (например, с версией 10.0.15063.0, также известной как пакет SDK для RS2). Однако в более поздних версиях заголовков Windows SDK (в частности в ShlObj.h и ShlObj_core.h) внесены исправления, чтобы это предупреждение не создавалось. При появлении этого предупреждения от заголовков Windows SDK можно выполнить следующие действия:

  1. Переключитесь на последнюю версию Windows SDK, поставляемую вместе с выпуском Visual Studio 2017 версии 15.5.

  2. Отключите предупреждение вокруг #include инструкции заголовка пакета Windows SDK:

   #pragma warning (push)
   #pragma warning(disable:4768)
   #include <shlobj.h>
   #pragma warning (pop)

Связывание extern constexpr

В более ранних версиях Visual Studio компилятор всегда использовал для переменной constexpr внутреннюю компоновку, даже если эта переменная была помечена как extern. В Visual Studio 2017 версии 15.5 новый параметр компилятора (/Zc:externConstexpr) обеспечивает корректное поведение, соответствующее стандартам. В конечном счете это поведение будет использоваться по умолчанию.

extern constexpr int x = 10;
error LNK2005: "int const x" already defined

Если файл заголовка содержит переменную, объявленную как extern constexpr, она должна быть помечена как __declspec(selectany) для правильного объединения повторяющихся объявлений:

extern constexpr __declspec(selectany) int x = 10;

typeid нельзя использовать с неполным типом класса

В более ранних версиях Visual Studio компилятор неправильно разрешал выполнение следующего кода, что приводило к выводу потенциально неверных сведений о типе. В Visual Studio 2017 версии 15.5 компилятор правильно выдает сообщение об ошибке:

#include <typeinfo>

struct S;

void f() { typeid(S); } //C2027 in 15.5
error C2027: use of undefined type 'S'

Тип целевого объекта std::is_convertible

Для std::is_convertible необходимо, чтобы целевой тип был допустимым возвращаемым типом. В более ранних версиях Visual Studio компилятор неправильно разрешал использование абстрактных типов, что могло приводить к неверному разрешению перегрузки и непредусмотренному поведению среды выполнения. Следующий код теперь правильно выводит предупреждение C2338:

#include <type_traits>

struct B { virtual ~B() = 0; };
struct D : public B { virtual ~D(); };

static_assert(std::is_convertible<D, B>::value, "fail"); // C2338 in 15.5

Чтобы избежать этой ошибки, при использовании is_convertible следует сравнить типы указателей, поскольку сравнение типа, отличного от указателя, может завершиться неудачно, если один тип является абстрактным:

#include <type_traits>

struct B { virtual ~B() = 0; };
struct D : public B { virtual ~D(); };

static_assert(std::is_convertible<D *, B *>::value, "fail");

Удаление спецификации динамических исключений и noexcept

В C++17 throw() является псевдонимом для noexcept, throw(<type list>) и throw(...) удалены, а определенные типы могут включать noexcept. Это изменение может привести к проблемам совместимости для исходного кода, который соответствует C++14 или более ранней версии. Параметр /Zc:noexceptTypes- можно использовать для возврата к noexcept версии C++14 при использовании режима C++17 в целом. Это позволяет обновить исходный код для соответствия C++17, не переписывая весь код throw().

Компилятор теперь также проверяет дополнительные спецификации несоответствующих исключений в объявлениях в режиме C++17 или с параметром /permissive- с новым предупреждением C5043.

Следующий код вызывает ошибки C5043 и C5040 в Visual Studio 2017 версии 15.5 при использовании параметра /std:c++17:

void f() throw(); // equivalent to void f() noexcept;
void f() {} // warning C5043
void g() throw(); // warning C5040

struct A {
    virtual void f() throw();
};

struct B : A {
    virtual void f() { } // error C2694
};

Чтобы исправить ошибки и продолжать использовать /std:c++17, добавьте параметр /Zc:noexceptTypes- в командную строку или обновите код для использования noexcept, как показано в следующем примере:

void f() noexcept;
void f() noexcept { }
void g() noexcept(false);

struct A {
    virtual void f() noexcept;
};

struct B : A {
    virtual void f() noexcept { }
};

Встроенные переменные

Статические элементы данных constexpr теперь являются неявными inline. Это означает, что объявления в пределах класса теперь являются их определениями. Использование внешних определений для элементов данных static constexpr избыточно и не рекомендуется. В Visual Studio 2017 версии 15.5 при применении параметра /std:c++17 следующий код теперь вызывает предупреждение C5041:

struct X {
    static constexpr int size = 3;
};
const int X::size; // C5041: 'size': out-of-line definition for constexpr static data member is not needed and is deprecated in C++17

Теперь предупреждение C4768 extern "C" __declspec(...) включено по умолчанию

Это предупреждение было добавлено в Visual Studio 2017 версии 15.3, но было отключено по умолчанию. В Visual Studio 2017 версии 15.5 предупреждение включено по умолчанию. Дополнительные сведения см. в разделе Новое предупреждение для атрибутов __declspec.

Заданные по умолчанию функции и __declspec(nothrow)

Компилятор ранее разрешал объявление установленных по умолчанию функций с __declspec(nothrow), если соответствующие базовые функции или функции-члены разрешали исключения. Это противоречит стандарту C++ и может привести к неопределенному поведению во время выполнения. Согласно стандарту, такие функции должны быть определены как удаленные, если имеется несовпадение спецификации исключений. В режиме /std:c++17 следующий код вызывает ошибку C2280:

struct A {
    A& operator=(const A& other) { // No exception specification; this function may throw.
        ...
    }
};

struct B : public A {
    __declspec(nothrow) B& operator=(const B& other) = default;
};

int main()
{
    B b1, b2;
    b2 = b1; // error C2280: attempting to reference a deleted function.
             // Function was implicitly deleted because the explicit exception
             // specification is incompatible with that of the implicit declaration.
}

Чтобы исправить этот код, удалите __declspec(nothrow) из функции по умолчанию или удалите = default и предоставьте определение функции вместе с любой необходимой обработкой исключений:

struct A {
    A& operator=(const A& other) {
        // ...
    }
};

struct B : public A {
    B& operator=(const B& other) = default;
};

int main()
{
    B b1, b2;
    b2 = b1;
}

noexcept и частичные специализации

При наличии noexcept в системе типов частичные специализации для сопоставления отдельных вызываемых типов могут мешать компиляции или выбору основного шаблона из-за отсутствия частичной специализации для указателей на функции noexcept.

В таких случаях может потребоваться добавить дополнительные частичные специализации для обработки указателей на функции noexcept и указателей noexcept на функции-члены. Эти перегрузки допустимы только в режиме /std:c++17 или более поздних версий. Если вам необходимо поддерживать обратную совместимость с C++14 и вы создаете код, который будут использовать другие разработчики, следует защитить эти новые перегрузки внутри директив #ifdef. Если вы работаете в автономном модуле, то вместо использования условий #ifdef можно просто выполнять компиляцию с параметром /Zc:noexceptTypes-.

Приведенный ниже код компилируется при использовании режима /std:c++14, но не при использовании /std:c++17 выдает ошибку C2027:

template <typename T> struct A;

template <>
struct A<void(*)()>
{
    static const bool value = true;
};

template <typename T>
bool g(T t)
{
    return A<T>::value;
}

void f() noexcept {}

int main()
{
    return g(&f) ? 0 : 1; // C2027: use of undefined type 'A<T>'
}

Следующий код компилируется при указании параметра /std:c++17, так как компилятор выбирает новую частичную специализацию A<void (*)() noexcept>:

template <typename T> struct A;

template <>
struct A<void(*)()>
{
    static const bool value = true;
};

template <>
struct A<void(*)() noexcept>
{
    static const bool value = true;
};

template <typename T>
bool g(T t)
{
    return A<T>::value;
}

void f() noexcept {}

int main()
{
    return g(&f) ? 0 : 1; // OK
}

Улучшения соответствия в версии 15.6

C++17. Основы работы с библиотеками V1

P0220R1 включает технические спецификации основ библиотек для C++17 в стандарт. Содержит обновления для <experimental/tuple>, <experimental/optional>, <experimental/functional>, <experimental/any>, <experimental/string_view>, <experimental/memory>, <experimental/memory_resource> и <experimental/algorithm>.

C++17. Улучшение вычета аргументов шаблона класса для стандартной библиотеки

P0739R0. Переместите adopt_lock_t в начало списка параметров, чтобы scoped_lock разрешил согласованное использование scoped_lock. Разрешите конструктору std::variant участвовать в разрешении перегрузки в большем числе случаев, чтобы поддерживать присваивание копированием.

Улучшения соответствия в версии 15.7

C++17: изменение наследующих конструкторов

P0136R1 указывает, что объявление using, которое указывает конструктор, теперь делает соответствующие конструкторы базового класса видимыми для инициализаций производного класса, поэтому можно не объявлять дополнительные конструкторы для производного класса. Это является изменением формулировки по сравнению с C++14. В Visual Studio 2017 версии 15.7 и более поздних в режиме /std:c++17 и более поздних версий может оказаться недопустимым или иметь иную семантику код, который нормально работает в C++14 с использованием наследуемых конструкторов.

В следующем примере показана реакция на событие в C++14:

struct A {
    template<typename T>
    A(T, typename T::type = 0);
    A(int);
};

struct B : A {
    using A::A;
    B(int n) = delete; // Error C2280
};

B b(42L); // Calls B<long>(long), which calls A(int)
          //  due to substitution failure in A<long>(long).

В следующем примере показана реакция на событие /std:c++17 в Visual Studio 15.7:

struct A {
    template<typename T>
    A(T, typename T::type = 0);
    A(int);
};

struct B : A {
    using A::A;
    B(int n)
    {
        //do something
    }
};

B b(42L); // now calls B(int)

Дополнительные сведения см. в разделе Конструкторы.

C++17: расширенная инициализация агрегата

P0017R1

Если конструктор базового класса не является открытым, но доступен производному классу, то в режиме /std:c++17 и более поздних версий в Visual Studio 2017 версии 15.7 больше нельзя использовать пустые фигурные скобки для инициализации объекта производного типа. В следующем примере показано поведение согласования в C++14:

struct Derived;
struct Base {
    friend struct Derived;
private:
    Base() {}
};

struct Derived : Base {};
Derived d1; // OK. No aggregate init involved.
Derived d2 {}; // OK in C++14: Calls Derived::Derived()
               // which can call Base ctor.

В C++17 Derived теперь считается агрегатным типом. Это означает, что инициализация Base через закрытый конструктор по умолчанию происходит непосредственно как часть расширенного правила агрегатной инициализации. Закрытый конструктор Base ранее вызывался через конструктор Derived, и это работало успешно благодаря объявлению дружественных отношений. В следующем примере показано поведение C++17 в Visual Studio версии 15.7 в режиме /std:c++17:

struct Derived;
struct Base {
    friend struct Derived;
private:
    Base() {}
};
struct Derived : Base {
    Derived() {} // add user-defined constructor
                 // to call with {} initialization
};
Derived d1; // OK. No aggregate init involved.
Derived d2 {}; // error C2248: 'Base::Base': cannot access
               // private member declared in class 'Base'

C++17: объявление параметров шаблона, не являющихся типом, автоматически

P0127R2

В режиме /std:c++17 компилятор теперь может выводить тип аргумента шаблона, не являющегося типом и объявленного с помощью auto:

template <auto x> constexpr auto constant = x;

auto v1 = constant<5>;      // v1 == 5, decltype(v1) is int
auto v2 = constant<true>;   // v2 == true, decltype(v2) is bool
auto v3 = constant<'a'>;    // v3 == 'a', decltype(v3) is char

В результате код C++14 может быть недопустимым или иметь другую семантику. Например, стали допустимыми некоторые перегрузки, которые ранее были недопустимыми. В следующем примере показан код C++14, который компилируется, поскольку вызов example(p) привязан к example(void*);. В Visual Studio 2017 версии 15.7 в режиме /std:c++17 оптимальным решением будет шаблон функции example.

template <int N> struct A;
template <typename T, T N> int example(A<N>*) = delete;

void example(void *);

void sample(A<0> *p)
{
    example(p); // OK in C++14
}

В следующем примере показан код C++17 в Visual Studio 15.7 в режиме /std:c++17:

template <int N> struct A;
template <typename T, T N> int example(A<N>*);

void example(void *);

void sample(A<0> *p)
{
    example(p); // C2280: 'int example<int,0>(A<0>*)': attempting to reference a deleted function
}

C++17: начальное преобразование строк (частичное)

P0067R5. Независимые от языкового стандарта функции низкого уровня для преобразования между целыми числами и строками и между числами с плавающей запятой и строками.

C++20. Избегайте ненужных распада (частично)

P0777R1. Проводит различие между концепцией "вырождения" и простым удалением константы или квалификаторов ссылки. Новый признак типа remove_reference_t заменяет decay_t в некоторых контекстах. В Visual Studio 2019 реализована поддержка remove_cvref_t.

C++17: параллельные алгоритмы

P0024R2. Технические спецификации параллелизма включены в стандарт с небольшими изменениями.

C++17: hypot(x, y, z)

P0030R1. Добавлены три новые перегрузки в std::hypot для типов float, double и long double, у каждой из которых есть три входных параметра.

C++17: <filesystem>

P0218R1. Включает технические спецификации файловой системы в стандарт с некоторыми изменениями формулировок.

C++17: математические специальные функции

P0226R1. Включает предыдущие технические спецификации для специальных математических функций в стандартный заголовок <cmath>.

C++17: руководства по вычетам для стандартной библиотеки

P0433R2. Обновляет до STL, чтобы воспользоваться преимуществами внедрения в C++17 P0091R3, который добавляет поддержку для выведения аргумента шаблона класса.

C++17: восстановление начальных строковых преобразований

P0682R1. Перемещение новых функций по простому преобразованию строк из P0067R5 в новый заголовок <charconv> и другие усовершенствования, включая изменение обработки ошибок для использования std::errc вместо std::error_code.

C++17: constexpr для char_traits (частично)

P0426R1. Изменения в функциях-членах length, compare и find типа std::traits_type, позволяющие использовать std::string_view в константных выражениях. (В Visual Studio 2017 версии 15.6 поддерживается только для Clang/LLVM. В версии 15.7 почти полная поддержка предоставляется и для ClXX.)

C++17: аргумент по умолчанию в шаблоне первичного класса

Это изменение поведения является предварительным условием P0091R3 - Template argument deduction for class templates.

Ранее компилятор игнорировал аргумент по умолчанию в шаблоне основного класса.

template<typename T>
struct S {
    void f(int = 0);
};

template<typename T>
void S<T>::f(int = 0) {} // Re-definition necessary

В Visual Studio 2017 версии 15.7 в режиме /std:c++17 аргумент по умолчанию не игнорируется:

template<typename T>
struct S {
    void f(int = 0);
};

template<typename T>
void S<T>::f(int) {} // Default argument is used

Разрешение имен зависимых объектов

Это изменение поведения является предварительным условием P0091R3 - Template argument deduction for class templates.

В следующем примере компилятор в Visual Studio 15.6 и более ранних версий разрешает D::type для B<T>::type в шаблоне основного класса.

template<typename T>
struct B {
    using type = T;
};

template<typename T>
struct D : B<T*> {
    using type = B<T*>::type;
};

Visual Studio 2017 версии 15.7 в режиме /std:c++17 требует указать ключевое слово typename в инструкции using из D. Без typename компилятор создает предупреждение C4346 ('B<T*>::type': dependent name is not a type) и ошибку C2061 (syntax error: identifier 'type'):

template<typename T>
struct B {
    using type = T;
};

template<typename T>
struct D : B<T*> {
    using type = typename B<T*>::type;
};

C++17. Атрибут [[nodiscard]] — повышение порога предупреждения

В Visual Studio 2017 версии 15.7 в режиме /std:c++17 уровень предупреждений для C4834 увеличивается с W3 до W1. Вы можете отключить это предупреждение путем приведения к void или путем передачи /wd:4834 компилятору.

[[nodiscard]] int f() { return 0; }

int main() {
    f(); // warning C4834: discarding return value
         // of function with 'nodiscard'
}

Список инициализации для базового класса конструктора шаблонов с переменным числом аргументов

В предыдущих версиях Visual Studio список инициализации для базового класса конструктора шаблонов с переменным числом аргументов мог не содержать аргументов шаблона (что недопустимо), и при этом не выдавалась ошибка. В Visual Studio 2017 версии 15.7 компилятор выдает ошибку.

В следующем примере кода в Visual Studio 2017 версии 15.7 возникает Ошибка C2614.

template<typename T>
struct B {};

template<typename T>
struct D : B<T>
{

    template<typename ...C>
    D() : B() {} // C2614: D<int>: illegal member initialization: 'B' is not a base or member
};

D<int> d;

Чтобы устранить ошибку, измените выражение B() на B<T>().

Агрегатная инициализация constexpr

Предыдущие версии компилятора C++ неправильно обрабатывали агрегатную инициализацию constexpr. Компилятор принимал недопустимый код, в котором выражение aggregate-init-list содержало слишком много элементов, и создавал для него неверный код объекта. Примером является следующий код:

#include <array>
struct X {
    unsigned short a;
    unsigned char b;
};

int main() {
    constexpr std::array<X, 2> xs = { // C2078: too many initializers
        { 1, 2 },
        { 3, 4 }
    };
    return 0;
}

В Visual Studio 2017 версии 15.7 с обновлением 3 и более поздних версий в предыдущем примере теперь происходит ошибка C2078. В приведенном ниже примере показано, как исправить код. При инициализации std::array, когда используются вложенные списки инициализации в скобках, укажите для внутреннего массива собственный вложенный список в скобках:

#include <array>
struct X {
    unsigned short a;
    unsigned char b;
};

int main() {
    constexpr std::array<X, 2> xs = {{ // note double braces
        { 1, 2 },
        { 3, 4 }
    }}; // note double braces
    return 0;
}

Улучшения соответствия в версии 15.8

typename для неквалифицированных идентификаторов

В режиме /permissive- компилятор больше не принимает ложные ключевые слова typename в неквалифицированных идентификаторах в определениях шаблонов псевдонимов. Следующий код теперь вызывает ошибку C7511:

template <typename T>
using  X = typename T; // C7511: 'T': 'typename' keyword must be 
                       // followed by a qualified name

Чтобы исправить эту ошибку, измените вторую строку на using X = T;.

__declspec() справа от определений шаблонов псевдонимов

Ключевое слово __declspec теперь запрещено указывать справа от определения шаблона псевдонима. Ранее компилятор принимал этот код, но игнорировал его. Это не приводило к появлению предупреждения о нерекомендуемом элементе при использовании псевдонима.

Вместо этого можно использовать стандартный атрибут C++ [[deprecated]]. Он учитывается в Visual Studio, начиная с выпуска 2017 версии 15.6. Следующий код теперь вызывает ошибку C2760:

template <typename T>
using X = __declspec(deprecated("msg")) T; // C2760: syntax error:
                                           // unexpected token '__declspec',
                                           // expected 'type specifier'`

Чтобы исправить эту ошибку, измените код на следующий (укажите атрибут перед знаком "=" определения псевдонима).

template <typename T>
using  X [[deprecated("msg")]] = T;

Диагностика двухэтапного поиска имен

Для двухэтапного поиска имен нужно, чтобы независимые имена, используемые в тексте шаблона, отображались в шаблоне во время определения. Ранее компилятор Microsoft C++ мог не искать ненайденное имя до момента создания экземпляров. Теперь ему требуется, чтобы независимые имена привязывались уже в тексте шаблона.

Ошибка может возникать при поиске в зависимых базовых классах. Ранее компилятор позволял использовать имена, определенные в зависимых базовых классах. Это связано с тем, что выполнялся поиск имен во время создания экземпляра при разрешении всех типов. Теперь такой код считается ошибкой. В этом случае вы можете принудительно использовать поиск переменной во время создания экземпляров, задав ей в качестве квалификатора тип базового класса или другим образом сделав ее зависимой, например с помощью указателя this->.

В режиме /permissive- следующий код теперь вызывает ошибку C3861:

template <class T>
struct Base {
    int base_value = 42;
};

template <class T>
struct S : Base<T> {
    int f() {
        return base_value; // C3861: 'base_value': identifier not found
    }
};

Чтобы исправить эту ошибку, измените инструкцию return на return this->base_value;.

Примечание.

В версиях библиотеки Boost.Python, предшествующих 1.70, существовало относящееся к MSVC обходное решение для опережающего объявления шаблона в unwind_type.hpp. В режиме /permissive-, начиная с Visual Studio 2017 версии 15.8 (_MSC_VER==1915), компилятор MSVC правильно выполняет поиск имен с учетом аргументов. Теперь процесс согласуется с другими компиляторами, что делает это обходное решение ненужным. Чтобы избежать ошибки C3861: 'unwind_type': identifier not found, обновите библиотеку Boost.Python.

Прямые объявления и определения в пространстве имен std

Стандарт C++ запрещает пользователю добавлять опережающие объявления и определения в пространство имен std. Добавление объявлений или определений в пространство имен std или в пространство имен, вложенное в std, теперь приводит к неопределенному поведению.

В дальнейшем корпорация Майкрософт изменит место определения для некоторых типов стандартной библиотеки. Это изменение нарушит работу существующего кода, который добавляет прямые объявления в пространство имен std. Новое предупреждение C4643 помогает выявить такие проблемы с источником. Предупреждение можно включить в режиме /default. По умолчанию оно отключено. Оно повлияет на программы, которые компилируются с параметром /Wall или /WX.

Следующий код теперь вызывает ошибку C4643:

namespace std {
    template<typename T> class vector;  // C4643: Forward declaring 'vector'
                                        // in namespace std is not permitted
                                        // by the C++ Standard`
}

Чтобы исправить эту ошибку, используйте директиву #include вместо опережающего объявления.

#include <vector>

Конструкторы, делегирующие самим себе

Стандарт C++ предполагает, что компилятор должен породить диагностическое сообщение, если делегирующий конструктор делегирует самому себе. Компилятор Microsoft C++ в режимах /std:c++17 и /std:c++latest теперь выдает ошибку C7535.

Без этой ошибки следующая программа скомпилируется, но создаст бесконечный цикл.

class X {
public:
    X(int, int);
    X(int v) : X(v){} // C7535: 'X::X': delegating constructor calls itself
};

Чтобы избежать этого, делегируйте в другой конструктор.

class X {
public:

    X(int, int);
    X(int v) : X(v, 0) {}
};

Постоянные выражения для offsetof

Макрос offsetof обычно реализовывался с помощью макроса, требующего reinterpret_cast. Такое использование недопустимо в контекстах, требующих константного выражения, но обычно компилятор Microsoft C++ разрешал его. Макрос offsetof, входящий в состав стандартной библиотеки, правильно использует встроенную конструкцию компилятора (__builtin_offsetof), но многие разработчики применяли эту особенность для определения собственных offsetof.

В Visual Studio 2017 версии 15.8 компилятор ограничивает области, в которых можно использовать операторы reinterpret_cast в режиме по умолчанию, чтобы поведение кода лучше соответствовало стандарту C++. В режиме /permissive- ограничения еще строже. Использование результата offsetof в тех случаях, когда требуются константные выражения, может привести к тому, что код вызовет предупреждение C4644 или C2975.

Следующий код вызывает ошибку C4644 в режимах /std:c++17, а также ошибку C2975 в режиме /permissive-:

struct Data {
    int x;
};

// Common pattern of user-defined offsetof
#define MY_OFFSET(T, m) (unsigned long long)(&(((T*)nullptr)->m))

int main()

{
    switch (0) {
    case MY_OFFSET(Data, x): return 0; // C4644: usage of the
        // macro-based offsetof pattern in constant expressions
        // is non-standard; use offsetof defined in the C++
        // standard library instead
        // OR
        // C2975: invalid template argument, expected
        // compile-time constant expression

    default: return 1;
    }
}

Чтобы исправить эту ошибку, используйте offsetof, определенный через <cstddef>.

#include <cstddef>

struct Data {
    int x;
};

int main()
{
    switch (0) {
    case offsetof(Data, x): return 0;
    default: return 1;
    }
}

Квалификаторы cv-qualifier в базовых классах, участвующих в раскрытии пакета

Компиляторы Microsoft C++ предыдущих версий не обнаруживали квалификаторы cv-qualifier в базовом классе, если он также участвовал в раскрытии пакета.

В Visual Studio 2017 версии 15.8 в режиме /permissive- следующий код вызывает ошибку C3770:

template<typename... T>
class X : public T... { };

class S { };

int main()
{
    X<const S> x; // C3770: 'const S': is not a valid base class
}

Ключевое слово template и вложенные описатели имен

В режиме /permissive- компилятор теперь требует, чтобы ключевое слово template предшествовало имени шаблона, если оно стоит после зависимого вложенного описателя имен.

В режиме /permissive- следующий код теперь вызывает ошибку C7510:

template<typename T> struct Base
{
    template<class U> void example() {}
};

template<typename T>
struct X : Base<T>
{
    void example()
    {
        Base<T>::example<int>(); // C7510: 'example': use of dependent
            // template name must be prefixed with 'template'
            // note: see reference to class template instantiation
            // 'X<T>' being compiled
    }
};

Чтобы исправить эту ошибку, добавьте ключевое слово template в инструкцию Base<T>::example<int>();, как показано в следующем примере.

template<typename T> struct Base
{
    template<class U> void example() {}
};

template<typename T>
struct X : Base<T>
{
    void example()
    {
        // Add template keyword here:
        Base<T>::template example<int>();
    }
};

Улучшения соответствия в версии 15.9

Порядок вычисления слева направо для операторов ->*, [], >> и <<

Начиная с C++17, операнды операторов ->*, [], >> и << должны обрабатываться в порядке слева направо. Есть два случая, в которых компилятор не может обеспечить этот порядок:

  • если одно из выражений операндов является объектом, передаваемым по значению, или содержит объект, передаваемый по значению;

  • при компиляции с использованием /clr, когда один из операндов является полем объекта или элемента массива.

Компилятор выдает предупреждение C4866, когда он не может обеспечить вычисление слева направо. Компилятор создает это предупреждение только в том случае, если указан режим /std:c++17 или более поздней версии, так как в C++17 добавлено требование обработки этих операторов в порядке слева направо.

Чтобы устранить это предупреждение, сначала определите, требуется ли вычисление операндов слева направо. Например, это может быть необходимо, если вычисление операндов может привести к побочным эффектам, зависящим от порядка. Во многих случаях порядок вычисления операндов не имеет никакого значения. Если порядок вычисления должен быть слева направо, попробуйте передать операнды с использованием ссылки на константу. Это изменение устраняет предупреждение в следующем примере кода.

// C4866.cpp
// compile with: /w14866 /std:c++17

class HasCopyConstructor
{
public:
    int x;

    HasCopyConstructor(int x) : x(x) {}
    HasCopyConstructor(const HasCopyConstructor& h) : x(h.x) { }
};

int operator>>(HasCopyConstructor a, HasCopyConstructor b) { return a.x >> b.x; }

// This version of operator>> does not trigger the warning:
// int operator>>(const HasCopyConstructor& a, const HasCopyConstructor& b) { return a.x >> b.x; }

int main()
{
    HasCopyConstructor a{ 1 };
    HasCopyConstructor b{ 2 };

    a>>b;        // C4866 for call to operator>>
};

Идентификаторы в шаблонах псевдонимов членов

Идентификатор, используемый в определении шаблона псевдонима члена, должен быть объявлен до использования.

В предыдущих версиях компилятора допускался следующий код. В Visual Studio 2017 версии 15.9 в режиме /permissive- компилятор выдает ошибку C3861:

template <typename... Ts>
struct A
{
  public:
    template <typename U>
    using from_template_t = decltype(from_template(A<U>{})); // C3861:
        // 'from_template': identifier not found

  private:
    template <template <typename...> typename Type, typename... Args>
    static constexpr A<Args...> from_template(A<Type<Args...>>);
};

A<>::from_template_t<A<int>> a;

Чтобы устранить эту ошибку, объявите from_template_t перед from_template.

Изменения модулей

В Visual Studio 2017 версии 15.9 компилятор выводит ошибку C5050, если на сторонах создания и потребления используются несогласованные параметры командной строки для модулей. В следующем примере показаны две проблемы:

  • на стороне потребления (файл main.cpp) не указан параметр /EHsc;

  • на стороне создания используется версия C++ /std:c++17, а на стороне потребления — /std:c++14.

cl /EHsc /std:c++17 m.ixx /experimental:module
cl /experimental:module /module:reference m.ifc main.cpp /std:c++14

В обоих случаях компилятор выдает ошибку C5050:

warning C5050: Possible incompatible environment while
importing module 'm': mismatched C++ versions.
Current "201402" module version "201703".

Компилятор также выдает ошибку C7536 при каждом изменении файла .ifc. Заголовок интерфейса модуля содержит хэш SHA2 содержимого. При импорте файл .ifc хэшируется и проверяется на соответствие хэшу в заголовке. Если они не совпадают, происходит ошибка C7536:

error C7536: ifc failed integrity checks.
Expected SHA2: '66d5c8154df0c71d4cab7665bab4a125c7ce5cb9a401a4d8b461b706ddd771c6'

Частичное упорядочение с использованием псевдонимов и невыведенных контекстов

Существует расхождение реализации в правилах частичного упорядочения, использующих псевдонимы в невыведенных контекстах. В следующем примере GCC и компилятор Microsoft C++ (в режиме /permissive-) выдают ошибку, тогда как Clang принимает код.

#include <utility>
using size_t = std::size_t;

template <typename T>
struct A {};
template <size_t, size_t>
struct AlignedBuffer {};
template <size_t len>
using AlignedStorage = AlignedBuffer<len, 4>;

template <class T, class Alloc>
int f(Alloc &alloc, const AlignedStorage<T::size> &buffer)
{
    return 1;
}

template <class T, class Alloc>
int f(A<Alloc> &alloc, const AlignedStorage<T::size> &buffer)
{
    return 2;
}

struct Alloc
{
    static constexpr size_t size = 10;
};

int main()
{
    A<void> a;
    AlignedStorage<Alloc::size> buf;
    if (f<Alloc>(a, buf) != 2)
    {
        return 1;
    }

    return 0;
}

В предыдущем примере выводится ошибка C2668:

partial_alias.cpp(32): error C2668: 'f': ambiguous call to overloaded function
partial_alias.cpp(18): note: could be 'int f<Alloc,void>(A<void> &,const AlignedBuffer<10,4> &)'
partial_alias.cpp(12): note: or       'int f<Alloc,A<void>>(Alloc &,const AlignedBuffer<10,4> &)'
        with
        [
            Alloc=A<void>
        ]
partial_alias.cpp(32): note: while trying to match the argument list '(A<void>, AlignedBuffer<10,4>)'

Расхождение реализаций связано с регрессией в формулировках стандарта C++. Для устранения ошибки 2235 потребовалось удалить часть текста, который позволял упорядочивать эти перегрузки. В текущем стандарте C++ отсутствует механизм для частичного упорядочения этих функций, поэтому они считаются неоднозначными.

В качестве обходного решения мы рекомендуем не применять частичное упорядочение. Вместо этого используйте SFINAE для удаления конкретной перегрузки. В следующем примере используется вспомогательный класс IsA, позволяющий удалять первую перегрузку, если Alloc является специализацией A:

#include <utility>
using size_t = std::size_t;

template <typename T>
struct A {};
template <size_t, size_t>
struct AlignedBuffer {};
template <size_t len>
using AlignedStorage = AlignedBuffer<len, 4>;

template <typename T> struct IsA : std::false_type {};
template <typename T> struct IsA<A<T>> : std::true_type {};

template <class T, class Alloc, typename = std::enable_if_t<!IsA<Alloc>::value>>
int f(Alloc &alloc, const AlignedStorage<T::size> &buffer)
{
    return 1;
}

template <class T, class Alloc>
int f(A<Alloc> &alloc, const AlignedStorage<T::size> &buffer)
{
    return 2;
}

struct Alloc
{
    static constexpr size_t size = 10;
};

int main()
{
    A<void> a;
    AlignedStorage<Alloc::size> buf;
    if (f<Alloc>(a, buf) != 2)
    {
        return 1;
    }

    return 0;
}

Недопустимые выражения и типы, не являющиеся литералами, в определениях функций на основе шаблона

Недопустимые выражения и типы, не являющиеся литералами, теперь правильно выявляются в определениях шаблонных функций, которые специализируются явным образом. Ранее такие ошибки не возникали для определения функции. Тем не менее недопустимое выражение или нелитеральный тип будут по-прежнему диагностироваться при вычислении как часть константного выражения.

В предыдущих версиях Visual Studio следующий код компилировался без предупреждений:

void g();

template<typename T>
struct S
{
    constexpr void f();
};

template<>
constexpr void S<int>::f()
{
    g(); // C3615 in 15.9
}

В Visual Studio 2017 версии 15.9 этот код возвращает ошибку C3615:

error C3615: constexpr function 'S<int>::f' cannot result in a constant expression.
note: failure was caused by call of undefined function or one not declared 'constexpr'
note: see usage of 'g'.

Чтобы избежать этой ошибки, удалите квалификатор constexpr из явного создания экземпляра функции f().

См. также

Соответствие стандартам языка Microsoft C/C++