• 【C++】引用做函数返回值时必须要注意


    先看一段代码:

    int& Add(int a, int b) {
    	int c = a + b;
    	return c;
    }
    
    int main() {
    	int& ret = Add(1, 2);
    	cout << ret << endl;
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    运行出来打印的值是没问题的。但这段代码没问题吗?

    这里引用做了函数返回值,

    首先将 Add(1, 2) 的返回值赋给 ret

    ret 是个引用,

    并没有实际空间,

    而这段代码是能编过的,

    所以 ret 会指向一块空间,

    这块空间给谁了呢?

    image-20220806221822623

    很显然,这块空间是 Add 函数开辟给 c 的。

    出了函数的作用域,

    再访问那块空间实际上就越界访问了,

    此时 ret 是界外的那块空间的别名,这样会有什么副作用呢?

    看下面一段代码的运行结果:

    image-20220806224257958

    分析一下:

    首先解释一下 ret1 = 7 的原因,

    因为 ret1 指向的是 Add 函数开辟给变量 c 的那块空间,

    当出了函数的作用域 c 就销毁了,

    此时 ret 还是那块空间的别名,

    销毁 c 的同时那块空间储存的值并没有被重置为随机值,

    所以一开始给的那段代码打印出来的结果没问题。

    但是,后来又调用了一次 Add 函数,

    此时并没有将返回值赋给其他变量。

    因为 ret1 只是一个别名,

    它改变了说明它指向的那块空间储存的值改变了,

    所以这两次调用 Add 函数所开辟的函数栈帧是没有变的,

    创建局部变量 c 的位置也没有变。

    第一次打印 ret1 的值是 7 没问题了,

    但为什么第二次打印就成了随机值?

    image-20220806224950254

    这是调用第一次打印函数时的汇编代码,

    当运行完黄标指向的那行之前时 ret1 的值就已经变了,

    但还不是打印出来的那个随机值。

    刚刚运行完的那一行调用了 call 指令,

    此时去调用的是运算符重载函数 operator<<

    函数调用要开辟栈帧,这里也不例外,

    ret1 的值改变说明它指向的那块空间成了函数栈帧的一部分,

    函数在执行过程中改变了那一块空间的值,

    至于为什么 ret1 第一次打印出来的值是正确的,

    这是因为在第一次改变 ret1 指向的那块空间之前就已经将 ret1 输出到屏幕上来了。

    而我们可以看到后面又调用了几次 call 语句,

    在后面开辟函数栈帧的过程中 ret1 指向的那块空间的值被多次改变,

    第二次输出的随机值只是被改变之后的一个值,

    ret1 之后可能还会再改变。


    所以,如果函数返回时,出了函数作用域,如果返回对象还未还给系统,则可以使用引用返回,如果已经还给系统了,则必须使用传值返回

  • 相关阅读:
    大数计算:e^1000/300!
    通过开源如何赚钱生存发展
    iNFTnews|一键生成数字藏品,VERTU Web3手机是未来吗?
    java计算机毕业设计在线招生系统源代码+系统+数据库+lw文档
    如何恢复被MALLOX勒索病毒加密的重要数据?
    springboot二手书籍线上回收网站java ssm-0401u
    elementUi自定义表头,根据判断显示不同的表头
    在开源项目中看到一个改良版的雪花算法,现在它是你的了。
    关于toString()方法
    基于html+css的图展示94
  • 原文地址:https://blog.csdn.net/Ye_Ming_OUC/article/details/126202902