我试图让 googletest 在我的 c++ 项目上运行,其中一部分涉及使用EXPECT_THROW(statement, expected_exception);
. 我正在使用 XCode 并选择了“Apple LLVM Compiler 3.0”。所有这些都在 Snow Leopard 10.6.8、XCode 4.2 上。
即使使用显式虚拟案例,我也无法通过任何这些测试EXPECT_THROW(throw std::runtime_error(), std::runtime_error);
自己扩展宏(从 gtest/internal/gtest-internal.h:1114 GTEST_TEST_THROW_)到
bool gtest_caught_expected = false;
try {
// GTEST_SUPPRESS_UNREACHABLE_CODE_WARNING_BELOW_(statement);
throw std::runtime_error("sigh");
}
// catch (expected_exception const&) {
catch (std::runtime_error const& e){
std::cout << "const ref caught" << std::endl;
gtest_caught_expected = true;
}
// added by me to check it wasn't a const& issue
catch (std::runtime_error e){
std::cout << "type caught" << std::endl;
gtest_caught_expected = true;
}
catch (...) {
//gtest_msg.value =
// "Expected: " #statement " throws an exception of type "
//#expected_exception ".\n Actual: it throws a different type.";
//goto GTEST_CONCAT_TOKEN_(gtest_label_testthrow_, __LINE__);
std::cout << "unknown caught" << std::endl;
}
然后在gdb中设置一个断点catch catch
并单步执行,我可以看到catch(runtime_errors)
被跳过,并且catch(...)
正在执行。如果我注释掉该catch(...)
块,则catch(std::runtime_error const& e)
执行正确的语句。
将我的编译器设置为“LLVM GCC 4.2”可以解决这个问题,但我想以 clang++ 为目标。
我单独的 EXPECT_THROW 测试用例在命令行上使用 clang++ 手动编译时有效,所以我认为它一定是一些深奥的 xcode 或 llvm 设置?或者也许 LLVM 是如何将我的 runtime_error 变成其他类型的?我尝试了 acatch throw
但可以在该上下文中获取任何类型信息。
有没有人经历过这种情况或有任何想法?
编辑:
所以我还与 libprofile_rt.dylib 以及编译器标志一起链接-fprofile-arcs -fprofile-coverage
。删除编译器标志-fprofile-arcs
消除了该问题。令人讨厌,因为它打破了我的报道报告。
(与 librpofile_rt.a 链接有同样的问题)
当然,我不是唯一看到这一点的人,因为 LLVM 据说使用 googletest 作为他们的测试用例?!
不确定我是否应该将其发布为答案,或者是否会有更多知识渊博的人出现并提供真正的解决方案。