cout顺序,i++和++i
先看下以下代码
#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
}
- 首先先明白运算符重载:
ostream & operator <<(ostream &stream, T &t)
返回的是一个ostream类型的引用,为什么要返回引用,先留着待会说。
- 然后,cout是什么,是一个ostream类的对象。
- 接着,cout<<a;的实质是cout对象调用其成员函数operator<<(),即cout.operator<<(a),返回一个ostream类型的引用
- 那么cout << a << f();其实就是cout.operator<<(a).operator<<(f())。(可以把cout.operator<<(a)看成一个新的cout)
- 问题来了,cout是从右到左读取参数,如果参数是函数,则先执行函数体,再将返回的值压栈,否则直接将值压栈,最后再将栈中的值输出来(见上述例子的第一个输出)
- 如果涉及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;)
- 那么同样,cout是从右到左读取参数,若遇到i++,则立即将i的值压栈(注意是值,而不是i的引用),再执行i+1;若遇到++i,则先执行i+1,再将i的引用压栈(而不是i的值,是引用!!!);若遇到i,则将i的引用压栈。
- 最后一个问题,为什么
ostream & operator <<(ostream &stream, T &t)
要返回一个引用类型呢?如果不返回引用类型而是返回一个ostream对象的话,那么cout << a << b这样的式子就会有问题,编译不会通过的,因为cout<<a返回了一个ostream的对象,而这个对象是一个“值”,而不是一个“变量”,它不能作为左值。(比如a = 1是正确的,而1 = 1是错误的,因为1是一个值,而不是变量,不能作为左值)。
- 至此,上面的代码为什么会输出这些奇怪的结果就一目了然了,值得吐槽的一点是,这些跟编译器有关,研究起来很没意思,Anyway,搞清楚了还是挺开心的^^
cout顺序,i++和++i的更多相关文章
- c++顺序表基本功能
头文件 #define LIST_MAX_SIZE 5#define LISTINCREMENT 2#include<assert.h>#include<string>temp ...
- C++中如何建立一个顺序表
准备数据 #define MAXLEN 100 //定义顺序表的最大长度 struct DATA { char key[10]; //结点的关键字 char name[20]; int age; }; ...
- C++ 数据结构学习一(顺序表)
//SequentialList.h 顺序表模板类 #ifndef SEQUENTIAL_LIST_HXX#define SEQUENTIAL_LIST_HXX using std::cout; us ...
- 数据结构C++实现代码-顺序表
参考:https://blog.csdn.net/ebowtang/article/details/43094041 //seqList.h// //包含顺序表中的声明// #include<i ...
- 线性表之顺序表C++实现
线性表之顺序表 一.头文件:SeqList.h //顺序线性表的头文件 #include<iostream> ; //定义顺序表SeqList的模板类 template<class ...
- C++实现顺序查找,折半查找,插值查找
1.顺序查找 从数组起始扫描到数组结尾,判断该索引数组是否和关键字相等,成功返回1 代码如下: //顺序查找 int seqSearch(int *array, int low, int high, ...
- STL之set(唯一且有顺序)
set作为一个容器也是用来存储同一数据类型的数据类型,并且能从一个数据集合中取出数据, 在set中每个元素的值都唯一,而且系统能根据元素的值自动进行排序.应该注意的是set中数元素的值不能直接被改变. ...
- 数据结构实验1:C++实现静态顺序表类
写了3个多小时,还是太慢了.太菜了! 图1 程序运行演示截图1 实验1 1.1 实验目的 熟练掌握线性表的顺序存储结构. 熟练掌握顺序表的有关算法设计. 根据具体问题的需要,设计出合理的表示数据的顺序 ...
- C++顺序表(模板总结)
C++顺序表(模板总结) 总结: 1.模板类的实质是什么:让程序员写出和类型无关的代码 2.模板的对象时什么:方法或者类 3.是对类中的一系列操作,提供一个不固定数据类型的方法 用模板做的类的时候要指 ...
随机推荐
- windows域相关
查看域角色: netdom query fsmo
- 用MyEclipse10.0远程连接Mysql数据库服务器
说明:本文档所有的操作均在满足以下条件的情况下操作, A.远程Linux服务器已经安装好MySQL数据库 B.本地电脑可以ping通远程服务器 C.已经成功安装了Myeclipse 一.下载mysql ...
- PHP-PHP常见错误
错误1:foreach循环后留下悬挂指针 在foreach循环中,如果我们需要更改迭代的元素或是为了提高效率,运用引用是一个好办法: $arr = array(1, 2, 3, 4); foreach ...
- 使用QQ互联登录应用
QQ登录集成插件简介 互联网应用越来越多,通常每一个应用都会要求用户注册登录,粗略估记一下,QQ,微博,微信,银行帐号.邮箱,招聘网站账户,淘宝帐号,支付宝帐号,公司OA帐号....粗略算一下,十几个 ...
- Linux yum操作无效的解决方法
1.没网,试着:ping www.baidu.com 如果显示没有连接的话,就说明没网,也就无法使用yum 命令. 2.ping通了的话,还是是用不了yum命令,说明是yum镜像没有了,那么就得下载一 ...
- python--文件处理范例
import os,os.path,string dir="D:\\curl\\data" if (os.path.exists(dir)==False): print " ...
- HIVE的transform函数的使用
Hive的TRANSFORM关键字提供了在SQL中调用自写脚本的功能,适合实现Hive中没有的功能又不想写UDF的情况.例如,按日期统计每天出现的uid数,通常用如下的SQL SELECT date, ...
- C++刷题——2830: 递归求1*1+2*2+3*3+……+n*n
Description 定义一个递归函数sum int sum(int n); //函数声明,返回12+22+32+--+n2的和 注意:sum为递归函数 Input 正整数n的值 Output 12 ...
- Atitit .h5文件上传
Atitit .h5文件上传 1. 上传原理1 2. Html1 3. Js2 4. uploadV2.js2 5. upServlet & FileUploadService {3 6. 注 ...
- atitit.企业管理----商业间谍策略的使用与防务
atitit.企业管理----商业间谍策略的使用与防务 1. 间谍的历史 2 1.1. 公元前10世纪,<旧约全书>中的<士师记>里讲述了参孙的故事是最早的间谍故事. 2 1. ...