《函数调用时参数、返回值所做的优化处理+编译器NRV优化》

时间:2023-02-16 22:22:24

摘要:
c++中创建对象以及销毁对象时,会自动调用对象的构造、析构函数(默认的或自定义的),因此,当我们在c++中调用函数时应该特别注意传参以及返回值时的优化处理,以减少构造、析构函数调用时的开销。同时编译器也会针对不同情况的函数调用,在不产生错误的情况下采用“NRV优化”(函数返回时,不产生临时变量)。


示例代码:

#include<iostream>
using namespace std;


class AA
{
public:
    AA(int a = 10)
        :_a(a)
    {
        cout << "AA(int a = 10)"<< endl;
    }
    AA(const AA& s)
        :_a(s._a)
    {
        cout << "AA(const AA& s)" << endl;
    }
    ~AA()
    {
        cout << "~AA()" << endl;
    }
private:
    int _a;
};

AA& Test(AA& s)
{
    return s;
}

int main()
{
    AA a(5);
    Test(a);
    return 0;
}

传值调用、传值返回:

《函数调用时参数、返回值所做的优化处理+编译器NRV优化》


传引用调用、引用返回:
《函数调用时参数、返回值所做的优化处理+编译器NRV优化》


NRV编译优化:

//题目:
//Test1中调用了___次AA的拷贝构造函数, ___次AA的赋值运算符函数的重载。
//Test2中调用了___次AA的拷贝构造函数, ___次AA的赋值运算符函数的重载。
//Test3中调用了___次AA的拷贝构造函数, ___次AA的赋值运算符函数的重载。
//
#include<iostream>
using namespace std;

class AA
{};
AA f(AA a)
{
    return a;
}
void Test1()
{
    AA a1;
    a1 = f(a1);
}
void Test2()
{
    AA a1;
    AA a2 = f(a1);
}

void Test3()
{
    AA a1;
    AA a2 = f(f(a1));
}
int main()
{
    Test1();
    Test2();
    Test3();
    return 0;
}

参考答案:

//Test1中调用了_2_次AA的拷贝构造函数, _1_次AA的赋值运算符函数的重载。
//Test2中调用了_3_次AA的拷贝构造函数, _0_次AA的赋值运算符函数的重载。
//Test3中调用了_5_次AA的拷贝构造函数, _0_次AA的赋值运算符函数的重载。

正确否?
标准答案:

//Test1中调用了_2_次AA的拷贝构造函数, _1_次AA的赋值运算符函数的重载。
//Test2中调用了_2_次AA的拷贝构造函数, _0_次AA的赋值运算符函数的重载。
//Test3中调用了_3_次AA的拷贝构造函数, _0_次AA的赋值运算符函数的重载。

详解:
《函数调用时参数、返回值所做的优化处理+编译器NRV优化》