bazı operasyonlarını uygulamak için taban ile türetilmiş downcasting kullanır. tınlamak-3.5 için (-std=c++1y
) göre, bu mahzun constexpr
işlevlerinde yasadışı olmalıdır:constexpr ve cRTP: derleyici anlaşmazlık
test.cpp:42:16: error: static_assert expression is not an integral constant expression
static_assert(e() == 0, "");
^~~~~~~~
test.cpp:11:26: note: cannot cast object of dynamic type 'const base<derived>' to type 'const derived'
const noexcept { return static_cast<const Derived&>(*this)(); }
GCC mutlu compiles the code. Peki kim haklı? Clang haklıysa, constexpr
işlevinde hangi C++ 14 kısıtlaması bu yayınlamayı yasa dışı yapar?
template <class Derived>
class base
{
public:
constexpr auto operator()()
const noexcept { return static_cast<const Derived&>(*this)(); }
};
class derived : public base<derived>
{
public:
constexpr auto operator()()
const noexcept { return 0; }
};
template <class A, class B>
class expr : public base<expr<A, B>>
{
const A m_a;
const B m_b;
public:
constexpr explicit expr(const A a, const B b)
noexcept : m_a(a), m_b(b) {}
constexpr auto operator()()
const noexcept { return m_a() + m_b(); }
};
template <class D1, class D2>
constexpr auto foo(const base<D1>& d1, const base<D2>& d2)
noexcept { return expr<base<D1>, base<D2>>{d1, d2}; }
int main()
{
constexpr auto d = derived{};
constexpr auto e = foo(d, d);
static_assert(e() == 0, "");
}
Askerin standart referanslar aradığını düşünüyorum. – Columbo
@Columbo Bazı bulmaya çalışacağım. Ancak genel bir kural olarak, çalışma zamanında tanımlanmamış davranışların çoğu (tümü?), “Constexpr” değerlendirmelerinde hatalardır. –
Kodun iyi olduğunu düşündüm, daha yakından bakmama izin ver – Columbo