Warum der C++ Compiler ermöglicht es, eine Funktion als constexpr zu erklären, die constexpr werden können?Warum wird der C++ Compiler ermöglicht es, eine Funktion als constexpr zu erklären, die nicht constexpr werden kann?
Zum Beispiel: http://melpon.org/wandbox/permlink/AGwniRNRbfmXfj8r
#include <iostream>
#include <functional>
#include <numeric>
#include <initializer_list>
template<typename Functor, typename T, size_t N>
T constexpr reduce(Functor f, T(&arr)[N]) {
return std::accumulate(std::next(std::begin(arr)), std::end(arr), *(std::begin(arr)), f);
}
template<typename Functor, typename T>
T constexpr reduce(Functor f, std::initializer_list<T> il) {
return std::accumulate(std::next(il.begin()), il.end(), *(il.begin()), f);
}
template<typename Functor, typename T, typename... Ts>
T constexpr reduce(Functor f, T t1, Ts... ts) {
return f(t1, reduce(f, std::initializer_list<T>({ts...})));
}
int constexpr constexpr_func() { return 2; }
template<int value>
void print_constexpr() { std::cout << value << std::endl; }
int main() {
std::cout << reduce(std::plus<int>(), 1, 2, 3, 4, 5, 6, 7) << std::endl; // 28
std::cout << reduce(std::plus<int>(), {1, 2, 3, 4, 5, 6, 7}) << std::endl;// 28
const int input[3] = {1, 2, 3}; // 6
std::cout << reduce(std::plus<int>(), input) << std::endl;
print_constexpr<5>(); // OK
print_constexpr<constexpr_func()>(); // OK
//print_constexpr<reduce(std::plus<int>(), {1, 2, 3, 4, 5, 6, 7})>(); // error
return 0;
}
Ausgang:
28
28
6
5
2
Warum Fehler in dieser Zeile: //print_constexpr<reduce(std::plus<int>(), {1, 2, 3, 4, 5, 6, 7})>(); // error
auch für C++ 14 und C++ 1z?
std::plus
-constexpr T operator()(const T& lhs, const T& rhs) const;
(da C++ 14) - constexpr: http://en.cppreference.com/w/cpp/utility/functional/plusconstexpr initializer_list();
(da C++ 14) - construcot voninitializer_list
ist constexpr: http://en.cppreference.com/w/cpp/utility/initializer_list/initializer_list
Warum erlaubt der Compiler das Markieren von reduce()
als constexpr
, aber reduce()
kann nicht als Vorlagenparameter verwendet werden Wenn alle Parameter zur Kompilierungszeit an reduce()
übergeben wurden?
Der gleiche Effekt für einige Compiler - die unterstützt C++ 14-std=c++14
:
- x86 GCC 7.0.0
-std=c++1z -O3
: http://melpon.org/wandbox/permlink/AGwniRNRbfmXfj8r - x86 gcc 4.9.2
-std=c++14 -O3
: https://godbolt.org/g/wmAaDT - x86 gcc 6.1
-std=c++14 -O3
: https://godbolt.org/g/WjJQE5 - x86 Klirren 3.5
-std=c++14 -O3
: https://godbolt.org/g/DSCpYv - x86 Klirren 3.8
-std=c++14 -O3
: - https://godbolt.org/g/orSrgH - x86 Visual C++ Sie sollten copy-paste-Code: http://webcompiler.cloudapp.net/
- ARM gcc 4.8.2, ARM64 gcc 4.8, gcc PowerPC 4.8, AVR 4.5.3 gcc - nicht unterstützt C + 14
-std=c++14
Für alle diese Fälle zusammenstellen OK, bis ungenutzte Linie: //print_constexpr<reduce(std::plus<int>(), {1, 2, 3, 4, 5, 6, 7})>(); // error
"nennen _der C++ Compiler_ "? Welcher Compiler? GCC? Clang? MSVC? Welche Version? –