在使用cout对象时要注意的一个小问题

代码如下:

 1 #include <iostream>
 2
 3 using std::cout;
 4 using std::endl;
 5
 6 // 交换x和y的值并返回两者中较大的值
 7 int swap(int &x, int &y);
 8
 9 int main()
10 {
11     int x = 1, y = 2;
12     cout << swap(x, y) << " " << x << " " << y << endl;
13
14     return 0;
15 }
16
17 int swap(int &x, int &y)
18 {
19     int temp = x;
20     x = y;
21     y = temp;
22
23     return (x > y ? x : y);
24 }

swap()函数的功能是: 交换x和y的值并返回两者中较大的值 . 按照我的思路, 程序应该输出 2 2 1   但是最后的结果竟然是: 2 1 2

调试一下程序, 查看对应的汇编代码:

 1    0x004016cd <+29>:    call   0x409810 <__main>
 2    0x004016d2 <+34>:    movl   $0x1,-0x1c(%ebp)
 3    0x004016d9 <+41>:    movl   $0x2,-0x20(%ebp)
 4    0x004016e0 <+48>:    mov    -0x20(%ebp),%ebx
 5    0x004016e3 <+51>:    mov    -0x1c(%ebp),%esi
 6    0x004016e6 <+54>:    lea    -0x20(%ebp),%eax
 7    0x004016e9 <+57>:    mov    %eax,0x4(%esp)
 8    0x004016ed <+61>:    lea    -0x1c(%ebp),%eax
 9    0x004016f0 <+64>:    mov    %eax,(%esp)
10    0x004016f3 <+67>:    call   0x40175c <swap(int&, int&)>
11    0x004016f8 <+72>:    mov    %eax,(%esp)
12    0x004016fb <+75>:    mov    $0x464fa0,%ecx
13    0x00401700 <+80>:    call   0x438df0 <_ZNSolsEi>
14    0x00401705 <+85>:    sub    $0x4,%esp
15    0x00401708 <+88>:    movl   $0x466064,0x4(%esp)
16    0x00401710 <+96>:    mov    %eax,(%esp)
17    0x00401713 <+99>:    call   0x458000 <_ZStlsISt11char_traitsIcEERSt13basic_ostreamIcT_ES5_PKc>
18    0x00401718 <+104>:    mov    %esi,(%esp)
19    0x0040171b <+107>:    mov    %eax,%ecx
20    0x0040171d <+109>:    call   0x438df0 <_ZNSolsEi>
21    0x00401722 <+114>:    sub    $0x4,%esp
22    0x00401725 <+117>:    movl   $0x466064,0x4(%esp)
23    0x0040172d <+125>:    mov    %eax,(%esp)
24    0x00401730 <+128>:    call   0x458000 <_ZStlsISt11char_traitsIcEERSt13basic_ostreamIcT_ES5_PKc>
25    0x00401735 <+133>:    mov    %ebx,(%esp)
26    0x00401738 <+136>:    mov    %eax,%ecx
27    0x0040173a <+138>:    call   0x438df0 <_ZNSolsEi>
28    0x0040173f <+143>:    sub    $0x4,%esp
29    0x00401742 <+146>:    mov    $0x0,%eax
30    0x00401747 <+151>:    jmp    0x401751 <main()+161>
31    0x00401749 <+153>:    mov    %eax,(%esp)
32    0x0040174c <+156>:    call   0x40c570 <_Unwind_Resume>
33    0x00401751 <+161>:    lea    -0xc(%ebp),%esp
34    0x00401754 <+164>:    pop    %ecx
35    0x00401755 <+165>:    pop    %ebx
36    0x00401756 <+166>:    pop    %esi
37    0x00401757 <+167>:    pop    %ebp
38    0x00401758 <+168>:    lea    -0x4(%ecx),%esp
39    0x0040175b <+171>:    ret    

可以发现, 程序执行的顺序是:

  把y的值放到距离栈底1c字节的位置.

  把x的值放到距离栈底20字节的位置.

  把x的值放到距离栈底4字节的位置.

  把y的值放到栈底

  然后再执行swap()

  然后再把距离栈底4字节的值(x = 1)输出

  然后再把栈底的值(y = 2)输出

所以, 内存数据区中的x和y的值确实是被交换了, 但是在作为cout的参数来输出的时候, 输出的是交换之前的值, 所以最后的结果就是 2 1 2

时间: 2024-12-29 16:55:12

在使用cout对象时要注意的一个小问题的相关文章

条款12:复制对象时请勿忘每一个成分

条款12:复制对象时请勿忘每一个成分 当为一个类实现自己的构造函数,相关赋值函数,析构函数,则必须有责任对类中的每一个成员进行初始化.赋值.释放.因此:如果为一个类添加一个成员,就必须同时相应修改上面几类函数. 看一个简单的类 class Terminal { Terminal(const int termid) : m_termId(termid) {} ~Terminal() {} Terminal(const Terminal & terminal) { this->m_termId

Effective C++ 条款11,12 在operator= 中处理&ldquo;自我赋值&rdquo; || 复制对象时不要忘记每一个成分

1.潜在的自我赋值     a[i] = a[j];     *px = *py; 当两个对象来自同一个继承体系时,他们甚至不需要声明为相同类型就可能造成别名. 现在担心的问题是:假如指向同一个对象,当其中一个对象被删,另一个也被删,这会造成不想要的结果. 该怎么办? 比如:   widget& widget:: operator+ (const widget& rhs) {    delete pd;    pd = new bitmap(*rhs.pb);    return *thi

复制对象时切记复制每一个成分

前言 标题一看似乎是非常直观简单,没必要特别写下一篇随便记录. 然而,在实际开发中,你会发现做到这一点对于经验不足的 C++ 程序员来说绝对是一个挑战. 要做到复制每一个成分,就一定小心下面说到的两点. 第一点:类中新增了成员变量之后,需要在自定义的构造函数,析构函数,赋值运算符重载函数,拷贝构造函数中做出相应调整. 请看下面一个例子: 1 // 记录调用信息 2 void logCall(const std::string & funcName); 3 4 // 顾客类 5 class Cus

使用Spring和SpringMVC管理bean时要注意的一个小细节

最近一直在做毕业设计...用到了Shiro和SpringMVC..用过shiro的朋友都知道shiro需要自己去写Realm,然后把Realm注入到SecurityManager中.而SecurityManager是shiro自带的类..我不可能去修改源码,增加注解,所以配置这个Realm bean的时候使用的是XML的配置方式..而又因为整个项目使用到了SpringMVC,我自己写的类基本都是用注解去配置的..然后就有了这么一个问题...我的问题与配置和http://www.oschina.n

使用 Nexus 搭建私服仓库时我犯的一个小错误

私服搭建好,啥都配置好了,纳闷的是 Repositories 中的 group 为何总是空值?我还反反复复删了又重建,结果还是一样,不经意间再看 Configuration 选项卡的内容,发现左右两个选项我弄反了:Ordered Group Repositories 里啥都木有,全部在 Available Repositories 里面,于是我尝试两边都要有内容,发现 public group 终于有了东西了.我骂了自己三遍文盲,然后认认真真地翻译了 “Ordered Group Reposit

C++返回值为对象时复制构造函数不执行怎么破

先说点背景知识,调用复制构造函数的三种情况: 1.当用类一个对象去初始化另一个对象时. 2.如果函数形参是类对象. 3.如果函数返回值是类对象,函数执行完成返回调用时. 在辅导学生上机时,有同学第3点提出异议.有教材上的例题为证: #include <iostream> using namespace std; class Point //Point 类的定义 { public: Point(int xx=0, int yy=0) { x = xx; //构造函数,内联 y = yy; } P

条款21:必须返回对象时,别妄想返回其reference

条款21:必须返回对象时,别妄想返回其reference 引用只是对象的一种别名当使用引用的时候,请确认他的另一个身份在哪? class Rational { public: Rational(int x, int y) : m_x(x), m_y(y){} //返回const是属于类型保护,friend修饰,以后条款详说 friend const Rational operator + (const Rational &lhs, const Rational &rhs) { Ration

[菜鸟]C++创建类对象时(无参)后不加括号与加括号的区别

在不考虑用new动态创建对象的情况下,用最普通的 类名 对象名(参数); 的方法创建对象 先贴上最正常最普通的代码 #include<iostream> using namespace std; class C{ public: C(){ cout<<"Hello C++\n"; }; C(int m){ cout<<"I love C++\n"; }; private: int n; }; int main(){ C c; re

在使用Linq to SQL并序列化一个数据对象的时候报System.InvalidOperationException异常,序列化类型XXX的对象时检测到循环引用。

在使用Linq to SQL并序列化一个数据对象的时候报System.InvalidOperationException异常,序列化类型 的对象时检测到循环引用. 异常信息(部分): System.Web.Services.Protocols.SoapException: 服务器无法处理请求. ---> System.InvalidOperationException: 生成 XML 文档时出错. ---> System.InvalidOperationException: 序列化类型 Web