zoukankan      html  css  js  c++  java
  • More Effective C++ 条款19 了解临时对象的来源

    1. 所谓的C++临时对象并不是程序员创建的用于存储临时值的对象,而是指编译器层面上的临时对象:这种临时对象不是由程序员创建,而是由编译器为了实现某些功能(例如函数返回,类型转换等)而创建.

        由于临时对象不是由程序员创建,其生存期由编译器掌控,因而也就不允许程序员对其进行更改,将其绑定到non-const 左值引用也就被禁止(见C++ 11: 右值引用,转移语义与完美转发),所有编译器都禁止将内置类型的non-const 左值引用绑定到内置类型的临时变量,但奇怪的是,有些编译器支持将类类型的non-const左值引用绑定到对应类类型的临时对象,比如vs.

    2. 临时对象通常在两种情况下被产生:一是当隐式类型转换在参数匹配时发生以使函数能够调用成功,二是函数返回对象的时候.了解这些临时对象那个何时被创建和销毁很重要,因为它们的构造和析构成本可能对程序性能造成影响.

        参数匹配过程中,编译器可能会产生临时对象:

    void  demo(const string& re);
    char arr[]="abdce";
    demo(arr);

        这里对demo的调用能够成功,编译器会执行一次隐式类型转换,调用string的拷贝构造函数构造一临时string对象并将arr的内容拷贝给它,然后将re绑定到该临时对象,当demo返回时,临时对象被销毁.

        值得注意的是,只有当demo的参数为const引用或按值传递时,上述对demo的调用才能奏效,加入demo的声明式如下:

    void demo(string& re);

        那么企图将arr作为参数调用demo会被编译器拒绝,理由正如1所言,不能将non-const左值引用绑定到临时对象,因为临时对象不受程序员掌控.

        函数返回时,也会产生临时对象:

    class Matrix;//Matrix是一个矩阵类型
    const Matrix operator+(const Matrix& lhs,const Matrix& rhs){
    Matrix sum;
    ...
    return sum;//sum存储最后结果
    }

        由于operator+调用结束时sum会被销毁,因此需要构造一临时Matrix对象,在return时将sum的值拷贝给该临时对象(具体过程可参考《深度探索C++面向对象模型》第二章 "返回值的优化").

    3. 临时对象可能很消耗资源,因此找出并(协助编译器)消除它们可能会给程序带来性能上的提升(具体见条款20,条款22).

  • 相关阅读:
    剑指Offer 07 重建二叉树
    剑指Offer 06 从尾到头打印链表
    剑指Offer 05 替换空格
    剑指Offer 04 二维数组中的查找
    剑指Offer 03 数组中重复的数字
    leetcode518
    leetcode474
    leetcode376
    leetcode646
    leetcode213
  • 原文地址:https://www.cnblogs.com/reasno/p/4833699.html
Copyright © 2011-2022 走看看