template <typename... Ts>
struct A
{
template <typename C>
constexpr auto proc() noexcept { return C{ }; }
constexpr size_t size() noexcept { return sizeof...(Ts); }
};
template <typename... Ts>
struct B : public A<Ts...>
{
using base_t = A<Ts...>;
template <typename... Cs>
constexpr auto example_prog_1() noexcept
{
constexpr size_t tmp = base_t::size();
// error C2131 : its not constexpr
return tmp + sizeof...(Cs);
}
template <typename... Cs>
constexpr auto example_prog_0() noexcept
{
return example_prog_1<decltype(base_t::template proc<Cs>())...>();
}
};
int main()
{
B<int, int, int, int> obj0 { };
constexpr size_t result0 = obj0.example_prog_0<char, char, char, char>();
}
это не работает. я получил ошибку C2131: Выражение не равно константе.
template <typename... Cs>
constexpr auto example_prog_1() noexcept
{
B dup{ };
constexpr size_t tmp = dup.size();
// error none : its fine
return tmp + sizeof...(Cs);
}
но сделать копию - это нормально.
он также работает, когда добавляется «это автоматическое сам» и используется.
может кто-нибудь сказать мне, почему? (msvc/std: c++ последняя версия)





base_t::size() это this->base_t::size().
Вам не разрешено разыменовывать указатели, которые были созданы вне константного выражения (или которые можно использовать в самих константных выражениях) в константном выражении. Это будет «чтение значения указателя», о котором компилятор не знает во время компиляции.
Это была формулировка до P2280R4 , которую MSVC не реализует. Эта статья и этот вопрос ( Почему ссылочный параметр в функции constexpr не является постоянным выражением?) Ответы более подробно объясняют, почему это произошло раньше.
Если у вас есть что-то, что не является указателем или ссылкой (например, по значению this auto self или B dup {};), это больше не применяется. Это постоянное выражение для вызова функции-члена, поскольку у вас нет доступа к объекту.
Самое простое решение — создать функцию sizestatic. Многие из ваших функций могут быть static.