先看下以下代码

#include<iostream>
using namespace std; int x = ; int f1()
{
x = ;
return x;
} int f2()
{
x = ;
return x;
} int main()
{
//Test1
cout << x << ' ' << f1() << ' ' << f2() << endl; //2 2 3 //Test2
int i = ; cout << i++ << ' ' << i << ' ' << ++i << endl; //1 2 2 //Test3
int s[] = {, , }; int *p = s; //cout << *p << ' ' << *(p++) << ' ' << *(++p) << endl; //3 2 3 //cout << *p++ << ' ' << *++p << ' ' << *p << endl; //2 3 3 //cout << *p << ' ' << *++p << ' ' << *p << endl; //2 2 2 //cout << *++p << ' ' << *p << ' ' << *++p << endl; //3 3 3 //cout << *p++ << ' ' << *p << ' ' << *p++ << endl; //2 3 1 //cout << *p << " " << *p++ << " " << *p << endl;//2 1 2
}
  1. 首先先明白运算符重载:

    ostream & operator <<(ostream &stream, T &t)

    返回的是一个ostream类型的引用,为什么要返回引用,先留着待会说。

  2. 然后,cout是什么,是一个ostream类的对象
  3. 接着,cout<<a;的实质是cout对象调用其成员函数operator<<(),即cout.operator<<(a),返回一个ostream类型的引用
  4. 那么cout << a << f();其实就是cout.operator<<(a).operator<<(f())。(可以把cout.operator<<(a)看成一个新的cout)
  5. 问题来了,cout是从右到左读取参数,如果参数是函数,则先执行函数体,再将返回的值压栈,否则直接将压栈,最后再将栈中的值输出来(见上述例子的第一个输出)
  6. 如果涉及i++和++i呢,问题就变得更加复杂。因为不同的编译器有不同的输出结果啊!这里选的是vs2013。这里先要明白什么是i++和++i
    T operator ++()                    //++var
    {
    var = var+;
    return var;
    }
    T operator ++(int dummy) //var++(dummy)
    {
    T tmp = var;
    var = var+;
    return tmp;
    }

    可以看到,++i比i++更有效率(少了一句T tmp=var;)

  7. 那么同样,cout是从右到左读取参数,若遇到i++,则立即将i的压栈(注意是值,而不是i的引用),再执行i+1;若遇到++i,则先执行i+1,再将i的引用压栈(而不是i的值,是引用!!!);若遇到i,则将i的引用压栈。
  8. 最后一个问题,为什么
    ostream & operator <<(ostream &stream, T &t)

    要返回一个引用类型呢?如果不返回引用类型而是返回一个ostream对象的话,那么cout << a << b这样的式子就会有问题,编译不会通过的,因为cout<<a返回了一个ostream的对象,而这个对象是一个“值”,而不是一个“变量”,它不能作为左值。(比如a = 1是正确的,而1 = 1是错误的,因为1是一个值,而不是变量,不能作为左值)。

  9. 至此,上面的代码为什么会输出这些奇怪的结果就一目了然了,值得吐槽的一点是,这些跟编译器有关,研究起来很没意思,Anyway,搞清楚了还是挺开心的^^

cout顺序,i++和++i的更多相关文章

  1. c++顺序表基本功能

    头文件 #define LIST_MAX_SIZE 5#define LISTINCREMENT 2#include<assert.h>#include<string>temp ...

  2. C++中如何建立一个顺序表

    准备数据 #define MAXLEN 100 //定义顺序表的最大长度 struct DATA { char key[10]; //结点的关键字 char name[20]; int age; }; ...

  3. C++ 数据结构学习一(顺序表)

    //SequentialList.h 顺序表模板类 #ifndef SEQUENTIAL_LIST_HXX#define SEQUENTIAL_LIST_HXX using std::cout; us ...

  4. 数据结构C++实现代码-顺序表

    参考:https://blog.csdn.net/ebowtang/article/details/43094041 //seqList.h// //包含顺序表中的声明// #include<i ...

  5. 线性表之顺序表C++实现

    线性表之顺序表 一.头文件:SeqList.h //顺序线性表的头文件 #include<iostream> ; //定义顺序表SeqList的模板类 template<class ...

  6. C++实现顺序查找,折半查找,插值查找

    1.顺序查找 从数组起始扫描到数组结尾,判断该索引数组是否和关键字相等,成功返回1 代码如下: //顺序查找 int seqSearch(int *array, int low, int high, ...

  7. STL之set(唯一且有顺序)

    set作为一个容器也是用来存储同一数据类型的数据类型,并且能从一个数据集合中取出数据, 在set中每个元素的值都唯一,而且系统能根据元素的值自动进行排序.应该注意的是set中数元素的值不能直接被改变. ...

  8. 数据结构实验1:C++实现静态顺序表类

    写了3个多小时,还是太慢了.太菜了! 图1 程序运行演示截图1 实验1 1.1 实验目的 熟练掌握线性表的顺序存储结构. 熟练掌握顺序表的有关算法设计. 根据具体问题的需要,设计出合理的表示数据的顺序 ...

  9. C++顺序表(模板总结)

    C++顺序表(模板总结) 总结: 1.模板类的实质是什么:让程序员写出和类型无关的代码 2.模板的对象时什么:方法或者类 3.是对类中的一系列操作,提供一个不固定数据类型的方法 用模板做的类的时候要指 ...

随机推荐

  1. ACE中静态实例管理方式

    ACE中的很多类使用了单例模式,为了便于管理单例对象,ACE使用了一个组件——ACE_Framework_Component来专门管理. 我们以ACE_Reactor这个单例类的创建和释放为例. 1. ...

  2. 关于wxpy,使用Python玩转微信的问题

    在github上下载了,安装了之后在idle上运行,好像是说Python不能上网.新手求助.现在问题已经解决,是ssl 证书的问题,不能用最新的 复制内容到剪贴板 代码: sudo pip unins ...

  3. C#Virtual和Override的几种组合

    情况1: class A{public void Show()} class B:A{public void Show()} 编译通过,有警告让在B的方法里添加new关键字,以便将A的方法隐藏 编译时 ...

  4. CentOS下yum常用命令

    1.自动搜索最快镜像插件:yum install yum-fastestmirror 2.更换163的源. 首先:备份/etc/yum.repos.d/CentOS-Base.repomv /etc/ ...

  5. [华为机试练习题]5.IP地址推断有效性

    题目 推断输入的字符串是不是一个有效的IP地址 具体描写叙述: 请实现例如以下接口 boolisIPAddressValid(constchar* pszIPAddr) 输入:pszIPAddr 字符 ...

  6. 网络结构----ISO/OSI七层网络模型和TCP/IP四层网络模型

    对等实体在一次交互作用中传送的信息单位称为协议数据单元.它包含控制信息和用户数据两部分. 上下层实体之间的接口称为服务訪问点. ISO/OSI參考模型分为:物理层.数据链路层.网络层.传输层.会话层. ...

  7. 解决国内gem不能用的问题

    转自:http://www.haorooms.com/post/gem_not_use 最近在安装SASS的时候,用到gem命令,但是运行出行如下错误! C:\Users\len>gem ins ...

  8. 【c#】装箱与拆箱

    从内存执行的角度来看,值类型的内存分配在线程的堆栈上,而引用类型的内存分配在托管堆上.因此从值类型向引用类型的转换,势必牵涉到数据的拷贝与指针引用等操作. 装箱操作,大致过程为:在托管堆中分配新对象的 ...

  9. jquery的input:type=file实现文件上传

    <!DOCTYPE html> <html> <head> <title>html5_2.html</title> <style> ...

  10. oracle 创建表空间及oracle 11g表空间之最大最小

    /*分为四步 *//*第1步:创建临时表空间 */create temporary tablespace emaoyi_temp tempfile 'D:\app\Administrator\prod ...