c++ 临时变量

C++的临时变量
 它们是被神所遗弃的孩子,没有人见过它们,更没有人知道它们的名字.它们命中注定徘徊于命运边缘高耸的悬崖和幽深的深渊之间,
用自己短暂的生命抚平了生与死之间的缝隙.譬如朝露,却与阳光无缘.是该为它们立一座丰碑的时候了,墓铭志上写着:我来了,我走了,我快乐过.

许多人对临时变量的理解仅仅限于:
 string temp;
 其实,从C++的观点来看,这根本就不是临时变量,而是局部变量.
 
 C++的临时变量是编译器在需要的时候自动生成的临时性变量,它们并不在代码中出现.但是它们在编译器生成的二进制编码中是存在的,
 也创建和销毁.在C++语言中,临时变量的问题格外的重要,因为每个用户自定义类型的临时变量都要出发用户自定义的构造函数和析构函数(如果用户提供了)
 
 又是该死的编译器!又该有人抱怨编译器总在自己背后干着偷偷摸摸的事情了.但是如果离开了编译器的这些工作,我们可能寸步难行.
 
 如果X是一个用户自定义的类型,有默认构造函数,拷贝构造函数,赋值运算函数,析构函数(这也是类的4个基本函数),那么请考虑以下代码:
 X get(X arg)
 {
  return arg;
 }
 
 X a;
 X b = get(a);
 即使是这么简单的代码也是很难实现的

让我们分析一下代码执行过程中发生了什么?
 首先我要告诉你一个秘密:对于一个函数来说,无论是传入一个对象还是传出一个对象其实都是不可能的.
 让一个函数传入或传出一个内置的数据类型,例如int,是很容易的,但是对于用户自定义类型得对象却非常的困难,因为编译器总得找地方为这些对象
 写上构造函数和析构函数,不是在函数内,就是在函数外,除非你用指针或引用跳过这些困难
 
 那么怎么办?在这里,编译器必须玩一些必要的小花招,嗯,其中的关键恰恰就是临时变量
 
 对于以对象为形参的函数:
  void foo(X x0)
 {
 }

X xx;
 foo(xx);

编译器一般按照以下两种转换方式中的一种进行转换
 1.在函数外提供临时变量
  void foo(X& x0)   //修改foo的声明为引用
 {
 }
 X xx;        //声明xx
 X::X(xx);      //调用xx的默认构造函数
 X __temp0;     //声明临时变量__temp0
 X::X(__temp0, xx); //调用__temp0的拷贝构造函数
 foo(__temp0);    //调用foo
 X::~X(__temp0);   //调用__temp0的析构函数
 X::~X(xx);     //调用xx的析构函数

2.在函数内提供临时变量
  void foo(X& x0)   //修改foo的声明为引用
 {
  X __temp0;     //声明临时变量__temp0
  X::X(__temp0, x0); //调用__temp0的拷贝构造函数
  X::~X(__temp0);   //调用__temp0的析构函数 
 }
 X xx;        //声明xx
 X::X(xx);      //调用xx的默认构造函数
 foo(xx);      //调用foo
 X::~X(xx);     //调用xx的析构函数
 
 无论是在函数的内部声明临时变量还是在函数的外部声明临时变量,其实都是差不多的,这里的含义是说既然参数要以传值的
 语意传入函数,也就是实参xx其实并不能修改,那么我们就用一个一摸一样临时变量来移花接木,完成这个传值的语意
 但是这样做也不是没有代价,编译器要修改函数的声明,把对象改为对象的引用,同时修改所有函数调用的地方,代价确实巨大啊,
 但是这只是编译器不高兴而已,程序员和程序执行效率却没有影响

对于以对象为返回值的函数:
  X foo()
 {
  X xx;
  return xx;
 }
 
 X yy = foo();

编译器一般按照以下方式进行转换
  void foo(X& __temp0) //修改foo的声明为引用
 {
  X xx;        //声明xx
  X::X(xx);      //调用xx的默认构造函数

__temp0::X::X(xx); //调用__temp0的拷贝构造函数
  X::~X(xx);     //调用xx的析构函数
 }
 
 X yy;         //声明yy
 X __temp0;      //声明临时变量__temp0
 foo(__temp0);     //调用foo
 X::X(yy, __temp0);  //调用yy的拷贝构造函数
 X::~X(__temp0);    //调用__temp0的析构函数
 X::~X(yy);      //调用yy的析构函数
 
 既然我们已经声明了yy,为什么还要紧接着声明__temp0,其实这里完全可以把yy和临时变量合一
 优化后,上面的代码看起来象这个样子:
 void foo(X& __temp0) //修改foo的声明为引用
 {
  X xx;        //声明xx
  X::X(xx);      //调用xx的默认构造函数

__temp0::X::X(xx); //调用__temp0的拷贝构造函数
  X::~X(xx);     //调用xx的析构函数
 }
 
 X yy;         //声明yy
 foo(yy);       //调用foo
 X::~X(yy);      //调用yy的析构函数
 
 嗯,怎么说呢,这算是一种优化算法吧,其实这各个技巧已经非常普遍了,并拥有一个专门的名称Named Return Value(NRV)优化
 NRV优化如今被视为标准C++编译器的一个义不容辞的优化操作(虽然其需求其实超出了正式标准之外)

除了以类为参数以外,如果参数的类型是const T&类型,这也可能导致临时变量
  void fun(const string& str)
  const char* name = "wgs";
 fun(name);
  嗯,还记得在const文档中的论述吗?对于这种特殊的参数类型,编译器是很乐意为你做自动转换的工作的,代价嘛,就是一个临时变量,
  不过如果是你自己去做,大概就只能声明一个局部变量了
 
 为什么函数和临时变量这么有缘,其实根本的原因在于对象传值的语意,这一个也是为什么C++中鼓励传对象地址的原因
 
 和函数的情况类似的,还有一大类情况是临时变量的乐土,那就是表达式
 string s,t;
 printf("%s", s + t);
 这里s+t的结果该放在什么地方呢?只能是临时变量中.
 
 这个printf语句带来了新的问题,那就是"临时变量的生命期"是如何的?
 对于函数的情况,我们已经看到了,临时变量在完成交换内容的使命后都是尽量早的被析构了,那么对于表达式呢?
 如果在s+t计算后析构,那么print函数打印的就是一个非法内容了,因此C++给出的规则是:
 临时变量应该在导致临时变量创建的"完整表达式"求值过程的最后一个步骤被析构
 什么又是"完整表达式"?简单的说,就是不是表达式的子表达式
 这条规则听起来很简单,但具体实现起来就非常的麻烦了,例如:
 X foo(int n)
 if (foo(1) || foo(2) || foo(3) )
 其中X中有operator int()转换,所以可以用在if语句中
 这里的foo(1)将产生一个临时变量1,如果这部分为false,foo(2)将继续产生一个临时变量,如果这部分也为false,foo(3)...
 一个临时变量的参数居然是和运行时相关的,更要命的是你要记住你到底产生了几个临时变量并在这个表达式结束的时候进行析构以小心的维护对象构造和析构的一致
 我猜想,这里会展开成一段复杂的代码,并加入更多的if判断才能搞定,呵呵,好在我不是做编译器的
 
 上面的规则其实还有两条例外:
 string s,t;
 string v = 1 ? s + t : s - t;
 这里完整表达式是?语句,但是在完整表达式结束以后临时变量还不能立即销毁,而必须在变量v赋值完成后才能销毁,这就是例外规则1:
 凡含有表达式执行结果的临时变量,应该存留到对象的初始化操作完成后销毁
 
 string s,t;
 string& v = s + t;
 这里s+t产生的临时变量即使在变量v的赋值完成后也不能销毁,否则这个引用就没用了,这就是例外规则2:
 如果一个临时变量被绑定到一个引用,这个临时变量应该留到这个临时变量和这个引用那个先超出变量的作用域后才销毁
 
 这篇文章可能有些深奥了,毕竟大多数内容来自于<<Inside The C++ Object Model>>
 那么就留下一条忠告:
 在stl中,以下的代码是错误的
 string getName();
 char* pTemp = getName().c_str();
 getName返回的就是一个临时变量,在把它内部的char指针赋值给pTemp后析构了,这时pTemp就是一个非法地址
 确实如C++发明者Bjarne Stroustrup所说,这种情况一般发生在不同类型的相互转换上
 
 在Qt中,类似的代码是这样的
 QString getName();
 char* pTemp = getName().toAscii().data();
 这时pTemp是非法地址
 
 希望大家不要犯类似的错误

时间: 2025-01-02 00:41:02

c++ 临时变量的相关文章

代码重构之以查询取代临时变量

意图 - 使得同一个类中的所有函数都可以获得这份信息,能够为这个类编写更清晰的代码 示例 /** * 以查询取代临时变量之前 * Created by luo on 2017/4/19. */ public class ReplaceTempWithQueryBefore { private double _quantity; private double _itemPrice; public double test() { double basePrice = _quantity * _ite

重构改善既有代码设计--重构手法04:Replace Temp with Query (以查询取代临时变量)

所谓的以查询取代临时变量:就是当你的程序以一个临时变量保存某一个表达式的运算效果.将这个表达式提炼到一个独立函数中.将这个临时变量的所有引用点替换为对新函数的调用.此后,新函数就可以被其他函数调用. 例子如下: double basePrice = _quantity*_itemPrice; if (basePrice > 1000) { return basePrice * 0.95; } else { return basePrice * 0.98; } 重构之后代码: if (BasePr

第2章 重新组织函数(3):引入解释性变量、分解临时变量和移除对参数的赋值

5. 引入解释性变量(Introduct Explaining Variable) //引入解释性变量 //重构前 if((platform.toUpperCase().indexOf("MAC") > -1) && (browser.toUpperCase().indexOf("IE") > -1) && wasInitialized() && resize > 0) { //do somethin

JAVA不用临时变量交换两个变量的值

交换两个变量的值,不适用临时变量 1 public class Test{ 2 public static void main(String[] args){ 3 int a = 1; 4 int b = 2; 5 // 请交换两个变量的值 6 } 7 } 交换两变量的值 加减实现交换 1 public class Test{ 2 public static void main(String[] args){ 3 int a = 1; 4 int b = 2; 5 // 请交换两个变量的值 6

两个数字交换(不使用临时变量)

#include<stdio.h> #include<stdlib.h> void swap(int* a, int* b)//普通交换 {  int tmp = *a;  *a = *b;  *b = tmp; } //不使用临时变量 void swap1(int* a, int* b)//使用加减法 {  *a = *a + *b;  *b = *a - *b;  *a = *a - *b; } void swap2(int *a, int *b)//使用异或 {  *a = 

C++函数返回引用、非引用以及临时变量的问题

C++中新增了引用类型,所以函数的返回值可以是引用类型.那么就会有人想问 返回引用类型与返回非引用类型有区别吗? 结论是显然的,而且有明显的区别.尤其初学者会很容易绕进去.让我们先看四个函数原型.以int类型来举例 (1) int fun(...) { return ....//后面跟的是一个引用 } 例如:int fun(int &a) { return a; } (2)int fun(...) { return....//后面跟的是一个非引用 } 例如:int  fun(int a) { r

重构改善既有代码设计--重构手法02:Inline Method (内联函数)&amp; 03: Inline Temp(内联临时变量)

Inline Method (内联函数) 一个函数调用的本体与名称同样清楚易懂.在函数调用点插入函数体,然后移除该函数. int GetRating() { return MoreThanfiveLateDeliverise() ? 2 : 1; } bool MoreThanfiveLateDeliverise() { return _numberOfLateLiveries > 5; } int GetRating() { return _numberOfLateLiveries > 5

代码重构之内联临时变量

意图 - 有一个临时变量,只被一个简单表达式赋值一次,而它妨碍了其他重构手法 示例 /** * 内联临时变量之前 * Created by luo on 2017/4/19. */ public class InlineTempBefore { Order anOrder = new Order(); public boolean test() { double basePrice = anOrder.basePrice(); return basePrice > 10; } } /** * 内

代码重构之分解临时变量

意图 如果临时变量承担多个责任,它就应该被替换(分解)为多个临时变量,每个变量只承担一个责任 示例 /** * Created by luo on 2017/4/24. */ public class SplitTemporaryVariableBefore { private double _height; private double _width; public void test(){ double temp = 2 * (_height + _width); System.out.pr

不使用临时变量 依次交换两个已知数组的值

刚开始编写中间使用了一个临时数组!!浪费空间!后面优化到只使用一个临时变量!最后到不使用临时变量! #include<stdio.h> int main() { int arr1[] = { 1, 2, 3 }; int arr2[] = { 4, 5, 6 }; int i = 0; int sz = sizeof arr1 / sizeof arr1[0]; for (i = 0; i < sz; i++) { arr1[i] = arr1[i] ^ arr2[i]; arr2[i]