std::launder是C++17引入的用于声明指针指向刚原地重建的有效对象的工具,它不改变指针值,仅向编译器认证对象生命周期合法性,防止因优化导致未定义行为。
std::launder 是 C++17 引入的一个极小但关键的工具,它不改变指针值,也不分配内存,而是向编译器“声明”:这个指针指向的对象,是**刚被原地重建(placement new)或以非常规方式构造出来的有效对象**。它的核心作用是**打破编译器对指针别名和对象生命周期的过度假设**,防止因激进优化导致未定义行为。
在 C++ 中,对象有明确的生命周期:构造开始、析构结束。编译器依赖这一模型做优化。比如,当你用 char buf[sizeof(T)] 申请原始内存,再用 new(buf) T{...} 原地构造一个 T,这块内存“现在”确实有了一个合法的 T 对象。但问题来了:
buf 指针“从来没见过 T”,它仍只把那块内存视为 char 数组;buf 强转为 T* 并解引用,C++ 标准规定这是未定义行为(UB),因为该指针并未指向一个通过标准方式“诞生”的 T 对象;T 对象根本不存在或没被修改。它不是魔法,而是一个显式契约:你告诉编译器,“请承认这个地址上现在有一个新活的对象”。典型用法如下:
char* 或 void*)转换为对应类型的指针;std::launder 包裹该指针,再使用返回值。示例:
char storage[sizeof(std::string)];它出现在几个底层模式中,不用就极易踩坑:
它不负责构造、不负责析构、不检查类型安全、不阻止 UB——它只解决“指针合法性认证”这一个窄问题:
std::launder(ptr) 之前没做 placement new → UB);