C++函数返回局部变量对象的优化-不调用复制构造函数(g++ -fno-elide-constructors选项)

先说点背景知识,调用复制构造函数的三种情况:

  1.当用类一个对象去初始化另一个对象时

  2.如果函数形参是类对象

  3.如果函数返回值是类对象,函数执行完成返回调用时

请看下面的例子:

#include <stdio.h>
#include <unistd.h>

#include <iostream>

class A
{
public:
    A() { std::cout << "A Constructor" << std::endl; }
    A(const A&) { std::cout << "A Copy Constructor" << std::endl; }
    A &operator=(const A&) { std::cout << "A operator = " << std::endl; }
    ~A() {std::cout << "A Destructor" << std::endl;}
};
class B
{
public:
    B() { std::cout << "B Constructor" << std::endl; }
    B(const B&) { std::cout << "B Copy Constructor" << std::endl; }
    ~B() {std::cout << "B Destructor" << std::endl;}
};

A getA(A ta, B tb)
{
    std::cout << "====-2" << std::endl;
    A a1;                                        //-----A Constructor
    std::cout << "====-2" << std::endl;
    return a1;                                    //-----A Copy Constructor temp=a1, A Destructor a1
}

int main()
{
    A a;                                        //----A Constructor
    B b;                                        //----B Constructor
    std::cout << "====-1" << std::endl;
    A a2 = getA(a,b);      //---- B Copy Constructor, A Copy Constructor.......(a2=temp) A Copy Constructor.....A Destructor(temp des)



   A a2;    //如果此处先定义了a2,然后再进行a2对象的再次赋值,会发生什么????而不是像上面那种直接进行定义赋值。结果见下面3图

   a2= getA(a,b);

    std::cout << "====-1" << std::endl;
    return 0;
}

编译:g++ -fno-elide-constructors moveConstruct.cpp -o moveConstruct

执行:./moveConstruct

结果:

1、这是我加了编译选项的结果: -fno-elide-constructors

编译:g++ moveConstruct.cpp -o moveConstruct

执行:./moveConstruct

结果:

2、这是我不加了编译选项的结果:

有两个问题???

A a1;  构造和析构发生在什么时候????

A a2 = getA(a,b); a2构造和析构发生在什么时候????

难道a1和a2合体了。。。

3、先定义了a2,然后再进行a2对象的再次赋值

“-fno-elide-constructors”选项起了作用,有图为证。下面中加上这个参数后,编译完看到的提示信息:

  

<think>好的,我需要解释QMAKE_CXXFLAGS中的-fno-elide-constructors选项的作用。首先,用户可能是在使用qmake构建系统时遇到了相关编译问题,或者想优化代码性能。这个选项是传递给编译器的标志,具体是g++或clang。需要先理解-fno-elide-constructors的含义。通常,编译器会进行返回优化(RVO)和命名返回优化(NRVO),这些优化可以省略必要的拷贝构造函数和移动构造函数调用,从而提高效率。而-fno-elide-constructors的作用是禁止这种优化,强制编译器执行所有构造函数调用,包括拷贝和移动构造。 用户可能想了解为什么需要禁用这个优化。常见场景包括调试时观察构造函数调用情况,或者在教学中展示拷贝/移动构造函数的行为。例如,在测试移动构造函数是否正确实现时,如果编译器优化了这些调用,可能导致无法观察到预期的效果,这时就需要禁用优化。 另外,结合用户提供的引用[1],移动构造函数类似于资源的转移,而是拷贝。如果编译器优化了构造过程,可能无法触发移动构造函数调用,从而影响对代码行为的理解。因此,使用-fno-elide-constructors可以确保移动构造函数被显式调用,方便调试和分析。 需要确保解释清晰,结构合理,同时提供使用该选项的典型场景和注意事项,比如性能影响。最后生成相关问题,帮助用户深入理解相关主题。</think>### QMAKE_CXXFLAGS中`-fno-elide-constructors`的作用解释 `-fno-elide-constructors`是传递给C++编译器的标志,用于**禁用编译器的返回优化(RVO)和命名返回优化(NRVO)**。在Qt项目的`.pro`文件中,通过`QMAKE_CXXFLAGS += -fno-elide-constructors`可将此标志添加到编译选项。 #### 核心作用: 1. **禁止编译器省略构造函数调用** 默认情况下,编译器会优化掉临时对象的拷贝/移动构造过程,直接构造目标对象。例如: ```cpp MyClass func() { return MyClass(); // 默认情况下,编译器会直接构造返回值,跳过多余的拷贝/移动步骤 } ``` 添加此标志后,编译器会强制保留所有构造函数调用(包括拷贝构造和移动构造)。 2. **调试与分析** 当需要观察拷贝构造函数或移动构造函数的具体调用次数时(例如验证移动语义是否生效),禁用优化可以清晰展示对象构造过程[^1]。 3. **教学与测试** 在演示C++对象生命周期或构造函数行为时,此标志能确保所有隐式构造步骤可见。 #### 典型应用场景: - **验证移动语义**:若未禁用优化,移动构造函数可能被完全省略,导致无法验证其实现是否正确。 - **性能对比**:通过对比优化前后的构造函数调用次数,分析RVO/NRVO对性能的影响。 - **调试构造逻辑**:当构造函数包含副作用(如日志输出)时,需确保所有调用可见。 #### 注意事项: - **性能影响**:禁用优化可能导致额外构造/析构调用,降低程序效率。 - **编译器兼容性**:此标志主要适用于GCC和Clang,MSVC的类似标志为`/Od`(禁用优化)。 --- ###
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

隨意的風

如果你觉得有帮助,期待你的打赏

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值