C++不允许在函数的try中使用static局部变量的原因

本文通过一个具体的代码示例探讨了静态对象初始化与异常处理之间的交互作用,特别是当静态对象位于try块中时可能出现的问题,并解释了如何解决由此产生的析构函数调用冲突。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在《INSIDE C++ MODEL》书中247页提到了如下一段话:

“shiyon给被静态初始化的objects,有一些缺点。例如,如果exceptiony handling被支持,那些objects将不能够被放置于Atry区段之内...."

想了一段时间才想明白这个问题,下面用一段代码来说明一下:

A.cpp


class A

{
}


int main()

{   

    try

    {

        static A obj;

        int i = 0;

        int c = 10 /i;  //throw exception

    }

   catch(...)

    {
    }

}

对于main函数中的A obj,应该都知道,编译器会针对obj做一个唯一性初始化

即:

A* pObj = 0;


int main()

{

    try

    {

        static A obj;

        (pObj == 0) ? 0: obj.::A(), pObj = &obj;

        int i = 0;

        int c = 10 /i;  //throw exception

    }

   catch(...)

    {

      (pObj == 0) ? 0 : obj::~A(), pObj =0;

    }   

    (pObj == 0) ? 0 : obj::~A(), pObj =0;

}

对于try-catch,有一个这样的特性,对于try中出现的局部变量,会在catch中生成其析构函数的调用,这个就会和main()末尾的析构重叠了。



<think>我们被要求回答关于C++14和C++17中constexpr函数的新特性和改进。根据提供的引用和之前的知识,我们可以总结如下: 首先,回顾引用内容: 引用[1]提到C++14对constexpr的增强:允许函数包含多个返回语句,支持循环和条件语句,以及更广泛的应用场景。 引用[2]提到从C++11开始,类的构造函数可以声明为constexpr,并且强调了constexpr和const的区别:constexpr确保编译时计算,而const保证。 引用[3]提到constexpr函数可以用来初始化constexpr变量,且这种函数必须足够简单以便编译时计算。 引用[4]展示了constexpr变量的声明和初始化,其中提到constexpr函数可以用于初始化。 现在,我们针对C++14和C++17中constexpr函数的新特性进行详细说明: ### 1. C++14中的constexpr函数增强 C++14对constexpr函数进行了大幅增强,使其更加灵活和强大: - **允许函数体内包含多个语句**:在C++11中,constexpr函数只能包含一个return语句(可以使用条件运算符和递归)。而C++14允许函数体内包含多个语句,包括局部变量声明、循环(如for、while)和条件语句(如if-else)[^1]。 - **支持局部变量**:可以在constexpr函数中声明局部变量,只要这些变量是字面类型(literal type)并且在初始化后被修改(即常量)[^1]。 - **允许修改局部变量**:在constexpr函数中,可以修改在函数作用域内声明的对象(即局部变量),只要该变量的生命周期在constexpr函数内开始和结束[^1]。 - **更宽松的返回类型**:返回类型再要求是字面类型,但必须是有效的常量表达式类型(例如,能是void类型)[^1]。 示例(C++14): ```cpp constexpr int factorial(int n) { int result = 1; // 允许局部变量 for (int i = 1; i <= n; ++i) { // 允许循环 result *= i; } return result; } constexpr int value = factorial(5); // 编译时计算,value=120 ``` ### 2. C++17中的constexpr函数增强 C++17进一步扩展了constexpr的能力: - **constexpr lambda表达式**:允许lambda表达式声明为constexpr(或使用constexpr作为函数参数)[^2]。 - **if constexpr**:引入编译时条件语句`if constexpr`,它允许在编译时根据条件选择同的代码分支,从而简化模板元编程[^2]。 - **扩展constexpr使用场景**:允许在constexpr函数使用更多的标准库函数(如`<array>`、`<tuple>`等),并且支持更多的编译时计算[^2]。 示例(C++17): ```cpp // constexpr lambda constexpr auto square = [](int x) { return x * x; }; constexpr int sq = square(5); // sq=25 // if constexpr template <typename T> constexpr auto get_value(T t) { if constexpr (std::is_pointer_v<T>) { return *t; // 如果T是指针类型,则解引用 } else { return t; // 否则直接返回 } } ``` ### 3. 总结对比 | 特性 | C++11 | C++14 | C++17 | |---------------------|-------|-------|-------| | 函数体多语句 | ❌ | ✅ | ✅ | | 循环和条件语句 | ❌ | ✅ | ✅ | | 局部变量 | ❌ | ✅ | ✅ | | 修改局部变量 | ❌ | ✅ | ✅ | | constexpr lambda | ❌ | ❌ | ✅ | | if constexpr | ❌ | ❌ | ✅ | 这些增强使得constexpr函数在编译时计算中更加强大,能够处理更复杂的逻辑,同时也促进了模板元编程的简化。 §§相关问题§§ 1. 如何在C++17中使用constexpr lambda表达式? 2. if constexpr如何简化模板元编程? 3. constexpr函数在编译时计算失败会怎样? 4. C++20中constexpr又有哪些新特性? [^1]: C++14允许constexpr函数包含多个语句、循环和条件语句,并支持局部变量。 [^2]: C++17引入constexpr lambda和if constexpr,扩展了编译时计算能力。</think>### C++14 和 C++17 中 constexpr 函数的新特性与改进 #### 🚀 C++14 中的主要增强 [^1][^3] 1. **函数体限制放宽**: - 允许包含多个返回语句(再限制为单一 return 语句) - 支持循环结构(如 `for`、`while`) - 允许使用局部变量 ```cpp constexpr int factorial(int n) { int result = 1; // C++14 允许局部变量 for (int i = 1; i <= n; ++i) { // C++14 允许循环 result *= i; } return result; } ``` 2. **条件语句支持**: - 允许使用完整的 `if-else` 分支(再限制三元运算符) ```cpp constexpr bool isEven(int n) { if (n % 2 == 0) return true; // C++14 允许多返回语句 else return false; } ``` 3. **返回类型扩展**: - 支持 `void` 返回类型 - 允许返回字面类型(Literal Type) #### ⚡ C++17 中的关键改进 [^2] 1. **`if constexpr` 编译时分支**: - 实现编译期条件判断,消除运行时开销 ```cpp template<typename T> constexpr auto getValue(T t) { if constexpr (std::is_pointer_v<T>) return *t; // 编译时分支选择 else return t; } ``` 2. **lambda 表达式支持**: - 允许将 lambda 声明为 `constexpr` ```cpp constexpr auto square = [](int x) { return x * x; }; constexpr int sq = square(5); // 编译时计算 ``` 3. **标准库扩展**: - STL 组件(如 `std::array`、`std::tuple`)支持 `constexpr` 操作 ```cpp constexpr std::array<int, 3> arr = {1, 2, 3}; constexpr int first = arr[0]; // 编译时访问 ``` 4. **内存分配支持**: - 允许在 `constexpr` 函数使用 `new` 和 `delete`(需在编译期释放) #### 🔍 核心改进对比 | 特性 | C++11 | C++14 | C++17 | |---------------------|---------------|-------------------------|--------------------------------| | **函数体复杂度** | 单一 return | 循环/多语句/局部变量 | 动态内存分配 | | **条件语句** | 仅三元运算符 | 完整 `if-else` | `if constexpr` 编译时分支 | | **Lambda 支持** | ❌ | ❌ | ✅ | | **STL 支持** | 极有限 | 部分容器 | 扩展容器和算法 | #### 💡 应用场景优化 1. **编译时计算**: ```cpp constexpr int fib(int n) { if (n <= 1) return n; return fib(n-1) + fib(n-2); // C++14 支持递归 } static_assert(fib(10) == 55); // 编译时验证 ``` 2. **类型安全接口**: ```cpp class Point { public: constexpr Point(int x, int y) : x_(x), y_(y) {} // C++11起支持 constexpr int x() const { return x_; } // C++14 支持成员函数 private: int x_, y_; }; constexpr Point p(3,4); // 编译时构造对象 ``` #### ⚠️ 注意事项 1. 所有操作必须在编译期完成,能包含 I/O 或动态内存分配(C++17 除外) 2. 递归深度受编译器限制(通常可通过编译选项调整) 3. C++20 进一步扩展了 `constexpr` 支持(如虚函数、`try-catch` 等) > 关键演进:从 **简单常量计算** (C++11) → **复杂函数逻辑** (C++14) → **元编程和STL集成** (C++17)[^1][^2]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值