我认为这可能会有用,但我没有看到在gcc的标准库中实现。
基本上,在我看来,STL实现可以添加断言来检测(在运行时)简单的错误,如越界访问。我猜测这些访问在标准中无论如何都是未定义的行为,因此打印错误消息并且 - 可选 - 中止将符合标准。
这些断言当然可以在编译时通过旧的NDEBUG标志或其他标志来关闭。
作为一个小例子,我希望这段代码能够中止:
#include <vector>
int main()
{
return std::vector<int>{1,2}[2];
}
顺便说一下,我完全清楚像valgrind这样的工具的存在,但这可能是一个奖励。此外,像valgrind这样的内存检查工具不能保证在向量缩小后检测某些错误,例如访问越界元素,因为实现可能不会立即重新分配底层内存。
答案 0 :(得分:7)
我认为这可能会有用,但我没有看到在gcc的标准库中实现。
你错了,尝试定义_GLIBCXX_DEBUG
,这会让你的例子中止:
/home/jwakely/gcc/5/include/c++/5.0.0/debug/vector:402:error: attempt to
subscript container with out-of-bounds index 2, but container only
holds 2 elements.
Objects involved in the operation:
sequence "this" @ 0x0x7fffb8b221a0 {
type = NSt7__debug6vectorIiSaIiEEE;
}
Aborted (core dumped)
这些断言当然可以在编译时通过旧的NDEBUG标志或其他标志来关闭。
这使用单独的机制而非assert()
和NDEBUG
的原因是因为检查会增加开销,因此仅在显式请求时启用它。这意味着用户可以继续使用assert()
进行自己的检查,而无需打开标准库检查的开销。由于用于跟踪迭代器有效性的其他数据成员,还有一些二进制兼容性问题需要注意libstdc ++的调试模式。有关详细信息,请参阅https://gcc.gnu.org/onlinedocs/libstdc++/manual/debug_mode.html。
Libc ++也有类似的检查,使用不同的宏,VC ++在调试版本中自动启用类似的检查。