C++简明分析临时对象是什么

目录
  • 一、初探临时对象
    • 1.问题
    • 2.思考
    • 3.答案
  • 二、编译器的行为
  • 三、小结

一、初探临时对象

1.问题

下面的程序输出什么?为什么?

下面编写程序进行实验:

#include <stdio.h>
class Test {
    int mi;
public:
    Test(int i) {
        mi = i;
    }
    Test() {
        Test(0);
    }
    void print() {
        printf("mi = %d\n", mi);
    }
};
int main()
{
    Test t;
    t.print();
    return 0;
}

输出结果如下:

程序意图:

  • 在 Test() 中以 0 作为参数调用 Test(int i)
  • 将成员变量 mi 的初始值设置为 0

运行结果:

  • 成员变量 mi 的值为随机值

2.思考

构造函数是一个特殊的函数

  • 是否可以直接调用?
  • 是否可以在构造函数中调用构造函数?
  • 直接调用构造函数的行为是什么?

3.答案

  • 直接调用构造函数将产生一个临时对象
  • 临时对象的生命周期只有一条语句的时间(过了这条 C++ 语句,临时对象将被析构而不复存在)
  • 临时对象的作用域只在一条语句中
  • 临时对象是 C++ 中值得警惕的灰色地带

可以将上面代码写成这样,避免临时对象:

#include <stdio.h>
class Test {
    int mi;
    void init(int i)
    {
        mi = i;
    }
public:
    Test(int i) {
       init(i);
    }
    Test() {
       init(0);
    }
    void print() {
        printf("mi = %d\n", mi);
    }
};
int main()
{
    Test t;
    t.print();
    return 0;
}

输出结果如下:

再来看一个程序,深刻体会一下临时对象:

#include <stdio.h>
class Test {
    int mi;
    void init(int i)
    {
        mi = i;
    }
public:
    Test(int i) {
        printf("Test(int i)\n");
        init(i);
    }
    Test() {
        printf("Test()\n");
        init(0);
    }
    void print() {
        printf("mi = %d\n", mi);
    }
    ~Test()
    {
        printf("~Test()\n");
    }
};
int main()
{
    printf("main begin\n");
    Test();
    Test(10);
    printf("main end\n");
    return 0;
}

输出结果如下:

这个程序很好的说明了临时对象的生命周期只有一条语句的时间(过了这条 C++ 语句,临时对象将被析构而不复存在)

二、编译器的行为

现代 C++ 编译器在不影响最终执行结果的前提下,会尽力减少临时对象的产生!!!

下面来看一个例子:

#include <stdio.h>
class Test
{
    int mi;
public:
    Test(int i)
    {
        printf("Test(int i) : %d\n", i);
        mi = i;
    }
    Test(const Test& t)
    {
        printf("Test(const Test& t) : %d\n", t.mi);
        mi = t.mi;
    }
    Test()
    {
        printf("Test()\n");
        mi = 0;
    }
    int print()
    {
        printf("mi = %d\n", mi);
    }
    ~Test()
    {
        printf("~Test()\n");
    }
};
Test func()
{
    return Test(20);
}
int main()
{
    //Test t(10); 等价于 Test t = Test(10);
    Test t = Test(10); // ==> Test t = 10;
    Test tt = func();  // ==> Test tt = Test(20); ==> Test tt = 20;
    t.print();
    tt.print();
    return 0;
}

输出结果如下:

注意两点:

  • 通过输出结果可以看到【编译器并没有按照生成临时对象,再用临时对象初始化 t 对象(其中涉及调用拷贝构造函数)】的步骤,因为现代的编译器都会尽力避免临时对象的产生。
  • Test t = Test(10); 等价于 Test t = 10; 写成Test t = 10; 可以杜绝临时对象的产生。因为临时对象的产生会带来性能上的问题,Test t = Test(10); 相当于调用了两次构造函数, 而 Test t = 10; 少调用一次函数,性能得到提升。

三、小结

  • 直接调用构造函数将产生一个临时对象
  • 临时对象是性能的瓶颈,也是 bug 的来源之一
  • 现代 C++ 编译器会尽力避开临时对象
  • 实际工程开发中需要人为的避开临时对象

到此这篇关于C++简明分析临时对象是什么的文章就介绍到这了,更多相关C++临时对象内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 详解C++ 临时量与临时对象及程序的相关优化

    一.临时量与临时对象 临时量: 内置类型生成的临时量是常量(临时量,寄存器带出来). 自定义类型生成的临时量是变量 ,在内存中. 隐式生成生成的临时量是常量 ,显式生成生成的临时量是变量 . 临时对象: 临时对象是系统临时分配的对象,在没主动声明所需对象而又使用其功能时产生的 显示对象:出现类型名 隐式对象:不出现类型名 注意: 临时对象的生存周期只在本条语句,临时对象一旦被引用,它的生存周期就和引用相同. 对象如何生成? 先分配内存 在调用构造函数初始化对象的成员变量  产生对象对象析构了 对

  • C++中临时对象的常见产生情况及其解决的方案

    目录 前言 1. 以值传递的方式给函数传参 验证临时对象的而外开销(1) 解决方案 2. 类型转换成临时对象 / 隐式类型转换保证函数调用成功 验证临时对象的而外开销(2) 解决方案 3. 函数返回对象时候 验证临时对象的而外开销(3) 解决方案 前言 在C++中很容易就写出一些代码,这些代码的特点就是偷偷的给你产生了一些临时对象,导致临时对象会调用拷贝构造函数,赋值运算符,析构函数,假如该对象还有继承的话,也会调用父类的拷贝构造函数,赋值运算赋函数等.这些临时对象所调用的函数,都是不必要的开销

  • 详解C++ 中的临时对象

    C++中临时对象(Temporary Object)又称无名对象.临时对象主要出现在如下场景. 1.建立一个没有命名的非堆(non-heap)对象,也就是无名对象时,会产生临时对象. Integer inte= Integer(5); //用无名临时对象初始化一个对象 2.构造函数作为隐式类型转换函数时,会创建临时对象,用作实参传递给函数. 例: class Integer { public: Integer(int i):m_val(i){} ~Integer(){} private: int

  • 深入c++中临时对象的析构时机的详解

    c++中,临时对象一旦不需要,就会调用析构函数,释放其占有的资源:而具名对象则是与创建的顺序相反,依次调用析构函数. c++源码: 复制代码 代码如下: class X  {public:   int i;   int j;   ~X() {}   X() {} }; int main() {    X x1;    X();    x1.i = 1;    X x2; } 对应的汇编码: 复制代码 代码如下: _main    PROC ; 11   : int main() { push  

  • c++ 临时对象的来源

    首先看下面一端代码: 复制代码 代码如下: #include <iostream> void swap( int &a,int &b) {     int temp;     temp=a;     a=b;     b=temp; } int main(int argc,char** argv) {     int a=1,b=2;     swap(a,b);     std::cout<<a<<"-----"<<b&

  • C++简明分析临时对象是什么

    目录 一.初探临时对象 1.问题 2.思考 3.答案 二.编译器的行为 三.小结 一.初探临时对象 1.问题 下面的程序输出什么?为什么? 下面编写程序进行实验: #include <stdio.h> class Test { int mi; public: Test(int i) { mi = i; } Test() { Test(0); } void print() { printf("mi = %d\n", mi); } }; int main() { Test t;

  • C++简明分析讲解布尔类型及引用

    目录 一.C++中的布尔类型 二.C++中的三目运算符 三.C++中的引用 四.总结 一.C++中的布尔类型 C++在C语言的基本类型系统之上增加了bool C++中的bool可取的值只有true和 false 理论上bool只占用一个字节 C++编译器会将非0值转换为true ,0值转换为false 注意: true代表真值,编译器内部用1来表示 false代表非真值,编译器内部用0来表示 下面看一下这段代码,加深一下对bool类型的理解. #include <stdio.h> int ma

  • C语言简明分析选择结构和循环结构的使用

    目录 一.程序中的选择结构(if) 二.switch多分支选择语句 三.while循环结构 四.do...while 与 for 五.break和continue 一.程序中的选择结构(if) if的用法 上代码: #include <stdio.h> int main() { short a = 1; short b = 2; int c = a - b; if( c > 0 ) { printf("a > b\n"); } else { printf(&qu

  • C++简明分析讲解引用与函数提高及重载

    目录 详解引用 引用的基本使用 引用做函数参数 引用做函数返回值 常量引用 引用的本质 函数提高 函数默认值 函数占位参数 函数重载及注意事项 详解引用 引用的基本使用 语法:数据类型 &新变量名 =原来变量名 作用:给变量起别名 注意事项: 1.引用必须初始化 2.一旦初始化就不能更改(具体原因在下面引用本质上会讲到) 示例: int a = 10; int c = 20; 如果写 int &b;这是错误的,没有初始化引用,编译器不知道b指向的地址. 所以这样写 int &b=a

  • C语言简明分析指针与引用的具体用法

    目录 1.指针 2.引用 1.指针 在计算机中,数据是存放在内存单元中的,一般把内存中的一个字节称为一个内存单元.为了更方便地访问这些内存单元,可预先给内存中的所有内存单元进行地址编号,根据地址编号,可准确找到其对应的内存单元.由于每一个地址编号均对应一个内存单元,因此可以形象地说一个地址编号就指向一个内存单元.C 语言中把地址形象地称作指针. 主要就是两个运算符:&和*. & 表示求地址,*表示求地址中的值,*也可以用来定义指针(int *p表示整型指针): int a=1; int *

  • C++简明分析inline函数的使用

    目录 inline函数 实例 要点 建议 inline函数 当程序执行函数调用时,系统要建立栈空间,保护现场,传递参数以及控制程序执行的转移等等, 这些工作需要系统时间和空间的开销. 请看如下程序段,读入一行字符串,逐个判断是否为数字字符: #include<iostream> using namespace std; bool IsNumber(char ch) { return ch>= ʹ0ʹ && ch <= ʹ9ʹ ? 1 : 0; } int main(

  • Spring简明分析Bean作用域

    目录 Bean作用域 一.singleton(单例模式) 二.protoType(原型模式) Bean作用域   经过前面的学习,我们可以知道bean是存在作用域的.   从spring的官方文档中发现spring支持六种作用域,我们只需要重点认识singleton.protoType即可,后面的作用域都是于web框架相关的. 一.singleton(单例模式)   就和图中的一样,如果bean的作用域为singleton,那么在IOC容器中只有每个bean只有一个唯一的实例被创建. 我们通过代

  • .NET单点登陆的实现方法及思路

    系统的基本架构 我们假设一个系统System包含Service客户服务中心.Shop网上购物中心和Office网上办公中心三个独立的网站. Service管理客户的资料,登录和注销过程.不论客户访问System的任何一个页面,系统都会转到登录界面,在用户登录后,系统会自动转会到客户上 次请求的页面.并且用户此后可以在System中无缝切换.不需要再次进行登录.即在System中实现单点登录SSO(Single Sign-On). 我们知道,用户的即时状态通常是使用Application.Sess

随机推荐