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>();
}
isso não funciona. recebi o erro C2131: a expressão não foi avaliada como uma constante.
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);
}
mas fazer uma cópia está bem.
também funciona ao adicionar 'this auto self' e usá-lo.
alguém poderia me dizer por quê? (msvc /std:c++mais recente)
base_t::size()
éthis->base_t::size()
.Você não tem permissão para desreferenciar ponteiros que foram feitos fora da expressão constante (ou que são utilizáveis nas próprias expressões constantes) em uma expressão constante. Isso seria "ler o valor do ponteiro", que o compilador não conhece durante a compilação.
Esse era o texto anterior ao P2280R4 , que o MSVC não implementa. As respostas desse artigo e desta pergunta ( Por que um parâmetro de referência em uma função constexpr não é uma expressão constante? ) explicam por que isso aconteceu antes com mais detalhes.
Quando você tem algo que não é um ponteiro ou uma referência (por exemplo, o valor por valor
this auto self
ouB dup {};
), isso não se aplica mais. É uma expressão constante para chamar a função membro porque você não acessa o objeto.A solução mais fácil é fazer a
size
funçãostatic
. Muitas de suas funções poderiam serstatic
.