vector::at() 在越界时抛 std::out_of_range 异常,operator[] 不检查越界、行为未定义;前者有运行时开销,后者零成本但危险;调试模式下可启用 operator[] 边界检查。
vector::at() 在索引超出 size() 范围时,抛出 std::out_of_range 异常;而 operato 完全不检查,行为是未定义的(UB)。这意味着用 
operator[] 访问 v[100] 时,哪怕 v.size() == 5,编译器不会报错、运行时也不一定崩溃——可能读到垃圾值,也可能意外改写相邻内存,甚至在某些优化级别下被编译器直接优化掉整个访问。
-D_GLIBCXX_DEBUG(GCC)或启用 MSVC 的 _ITERATOR_DEBUG_LEVEL=2,能让 operator[] 也带检查,但这仅限 debug 构建,不可依赖at() 的开销略高:每次调用都需比较索引与 size(),并准备异常处理路径;operator[] 是纯指针偏移,零成本抽象for (size_t i = 0; i ),用 operator[] 更自然;若索引来自用户输入、文件解析或计算结果,优先用 at()
无论是栈上数组 int a[10]; 还是堆上 new int[10],C++ 标准完全不强制任何越界检查。访问 a[15] 或 p[20] 就是未定义行为——不报错、不中断、不警告(除非静态分析工具介入)。编译器可能生成看似“正常”的代码,但结果不可预测。
-fsanitize=address(ASan)和 GCC 的同名选项可在运行时捕获大部分越界读写,但会显著拖慢执行、增加内存占用,仅用于测试sizeof(a)/sizeof(a[0]) 来“安全”遍历函数参数里的数组——传入函数后,它退化为指针,sizeof 失效std::array 是固定大小的容器,其 operator[] 和原生数组一样不检查边界;但它的 at() 成员函数提供和 vector::at() 一致的异常检查语义。区别在于:std::array::at() 的检查在编译期已知大小,部分优化器可能更激进地内联或消除冗余判断,但标准不保证。
std::array + operator[] 是零开销替代原生数组的安全升级operator[] 和 at();同一段逻辑里保持检查策略一致,避免漏掉某处std::array 没有 data() 以外的动态分配,所以 at() 异常开销比 vector 略低,但差异微乎其微真正棘手的不是程序立刻 segfault,而是越界写入恰好覆盖了 nearby 变量、虚表指针或 malloc 元数据,导致几秒甚至几分钟后才崩,且崩溃点与问题源头毫无关联。这类 bug 在多线程环境下更难复现,ASan 有时也抓不到(比如只读越界未触发保护页)。
at() 是唯一可依赖的运行时防护手段,但它不能替代逻辑校验assert(i ,比异常更轻量,且 release 版本可通过 NDEBUG 移除
at(),若在循环中 push_back() 导致 size() 增长,原索引可能仍合法;但若 erase() 或 clear() 后继续用旧索引,at() 就会如期抛异常——这反而是好事