如何让 `static_assert` 打印它在失败时得到的值?
How to make the `static_assert` print the value it got on failure?
运行这个:
#define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__)
template< int PathLength >
constexpr const int startfindlastslash(const char (&path)[PathLength]) {
return PathLength;
}
int main(int argc, char const *argv[])
{
STATIC_ASSERT( startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 );
}
你得到了:
g++ -o main.exe --std=c++14 test_debugger.cpp
test_debugger.cpp: In function ‘int main(int, const char**)’:
test_debugger.cpp:1:28: error: static assertion failed: startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11
#define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__)
^
test_debugger.cpp:10:5: note: in expansion of macro ‘STATIC_ASSERT’
STATIC_ASSERT( startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 );
^~~~~~~~~~~~~
clang++ -Xclang -ast-print -fsyntax-only --std=c++14 test_debugger.cpp > main.exe
test_debugger.cpp:10:5: error: static_assert failed due to requirement 'startfindlastslash("cppdebugger/test_debugger.cpp") == 11' "startfindlastslash(
\"cppdebugger/test_debugger.cpp\" ) == 11"
STATIC_ASSERT( startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 );
^ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
test_debugger.cpp:1:28: note: expanded from macro 'STATIC_ASSERT'
#define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__)
^ ~~~~~~~~~~~
1 error generated.
编译器不会告诉它应该得到哪个值,它只说值不相等。
[ 这并不是对 "How to make static_assert print the value" 的真正回答,而是对 OP 为什么会出现他现在的行为的解释。 ]
C++17 之前的版本,static_assert
有两个参数。
第一个是要计算的表达式,第二个是当表达式计算结果不为真时要打印的消息。
当您定义 STATIC_ASSERT
宏时,您就是在向 static_assert
提供您想要打印的消息 - 它正在为您打印该消息。
我经常发现将 static_asserts
包装在函数模板中很有用。然后编译器将至少打印错误信息中的函数模板参数。如果将此策略与 @Phil1970 的价值模板方法结合使用,您将获得相当不错的结果。
使用 GCC 编译时,以下代码片段会打印包含您的值的错误消息:In instantiation of 'constexpr void assert_equal(const_val<T, A>, const_val<U, B>) [with T = int; T A = 30; U = int; U B = 11]'
Clang 更好:<source>:13:5: error: static_assert failed due to requirement '30 == 11' "Values are not equal!"
.
#define MAKE_CONST(x) const_val<decltype(x), x>{}
#define STATIC_ASSERT_EQUAL(x, y) assert_equal(MAKE_CONST(x), MAKE_CONST(y));
template<typename T, T val>
struct const_val {
constexpr const_val() = default;
constexpr const_val(T v) {}
};
template<typename T, T A, typename U, U B>
constexpr void assert_equal(const_val<T, A>, const_val<U, B>) {
static_assert(A == B, "Values are not equal!");
}
template< int PathLength >
constexpr const int startfindlastslash(const char (&path)[PathLength]) {
return PathLength;
}
int main(int argc, char const *argv[])
{
STATIC_ASSERT_EQUAL(startfindlastslash( "cppdebugger/test_debugger.cpp" ), 11);
}
这不完全是您使用的语法,为了方便起见,它涉及一个额外的宏,但希望它足以满足您的目的...
实时代码here.
运行这个:
#define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__)
template< int PathLength >
constexpr const int startfindlastslash(const char (&path)[PathLength]) {
return PathLength;
}
int main(int argc, char const *argv[])
{
STATIC_ASSERT( startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 );
}
你得到了:
g++ -o main.exe --std=c++14 test_debugger.cpp
test_debugger.cpp: In function ‘int main(int, const char**)’: test_debugger.cpp:1:28: error: static assertion failed: startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 #define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__) ^ test_debugger.cpp:10:5: note: in expansion of macro ‘STATIC_ASSERT’ STATIC_ASSERT( startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 ); ^~~~~~~~~~~~~
clang++ -Xclang -ast-print -fsyntax-only --std=c++14 test_debugger.cpp > main.exe
test_debugger.cpp:10:5: error: static_assert failed due to requirement 'startfindlastslash("cppdebugger/test_debugger.cpp") == 11' "startfindlastslash( \"cppdebugger/test_debugger.cpp\" ) == 11" STATIC_ASSERT( startfindlastslash( "cppdebugger/test_debugger.cpp" ) == 11 ); ^ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ test_debugger.cpp:1:28: note: expanded from macro 'STATIC_ASSERT' #define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__) ^ ~~~~~~~~~~~ 1 error generated.
编译器不会告诉它应该得到哪个值,它只说值不相等。
[ 这并不是对 "How to make static_assert print the value" 的真正回答,而是对 OP 为什么会出现他现在的行为的解释。 ]
C++17 之前的版本,static_assert
有两个参数。
第一个是要计算的表达式,第二个是当表达式计算结果不为真时要打印的消息。
当您定义 STATIC_ASSERT
宏时,您就是在向 static_assert
提供您想要打印的消息 - 它正在为您打印该消息。
我经常发现将 static_asserts
包装在函数模板中很有用。然后编译器将至少打印错误信息中的函数模板参数。如果将此策略与 @Phil1970 的价值模板方法结合使用,您将获得相当不错的结果。
使用 GCC 编译时,以下代码片段会打印包含您的值的错误消息:In instantiation of 'constexpr void assert_equal(const_val<T, A>, const_val<U, B>) [with T = int; T A = 30; U = int; U B = 11]'
Clang 更好:<source>:13:5: error: static_assert failed due to requirement '30 == 11' "Values are not equal!"
.
#define MAKE_CONST(x) const_val<decltype(x), x>{}
#define STATIC_ASSERT_EQUAL(x, y) assert_equal(MAKE_CONST(x), MAKE_CONST(y));
template<typename T, T val>
struct const_val {
constexpr const_val() = default;
constexpr const_val(T v) {}
};
template<typename T, T A, typename U, U B>
constexpr void assert_equal(const_val<T, A>, const_val<U, B>) {
static_assert(A == B, "Values are not equal!");
}
template< int PathLength >
constexpr const int startfindlastslash(const char (&path)[PathLength]) {
return PathLength;
}
int main(int argc, char const *argv[])
{
STATIC_ASSERT_EQUAL(startfindlastslash( "cppdebugger/test_debugger.cpp" ), 11);
}
这不完全是您使用的语法,为了方便起见,它涉及一个额外的宏,但希望它足以满足您的目的...
实时代码here.