拷贝构造函数的调用
拷贝构造函数会在以下三中情况下被调用
(1)当类的一个对象去初始化该类的另一个对象时
int main(){
  Point a(1,2);
  Point b(a);//用对象a初始化对象b,拷贝构造被调用
  Point c=a;//用对象a初始化对象c,拷贝构造被调用
  return 0;
}
细节:上面两种只是写法形式上不一样,执行的操作完全一样
(2)如果函数的形参是类的对象,调用函数时,进行形参和实参的结合
void(Point p){
  cout<<p.getX()<<endl;
}
int main(){
  Point a(1,2);
  f(a);//函数的形参为类的对象,当调用函数时,拷贝构造函数被调用
  return 0;
}
细节:只有对象用值传递时,才会调用拷贝构造,就像上面的那样。如果用传递引用,则不会调用拷贝构造,即Point &p就不会拷贝构造,这样也会减少时间的调用,效率也会比较高,多采用这种形式。
(3)如果函数的返回值是类的对象,函数执行完成返回调用者时
Point g(){
  Point a(1,2);
  return a;//函数的返回值是类的对象,返回一个对象时会调拷贝构造
}
int main(){
  Point b;
  b=g();
  return 0;
}
细节:表面上函数g将a返回给了主函数,但是a是g()的局部对象,离开建立它的函数g以后就消亡了,不可能在返回主函数后继续生存。所以在处理这种情况时编译系统在主函数中创建一个无名临时对象,该临时对象的生存期只在函数调用处的表达式中。也就是表达式“b=g()”中,执行语句"return a;时,实际上是调用拷贝构造将a的值复制到临时对象中。函数g运行结束时对象a消失,但临时对象会存在于表达式"b=g()”中。计算完这个表达式后,临时对象的使命也就完成了,该临时对象便自动消失。

这里第三种情况有点特别会生成临时对象,接下来就是throw和return这里的相似的特性。

throw可以抛出的类型是 int、float、bool 等基本类型,也可以是指针、数组、字符串、结构体、类等聚合类型。

看一个习题
#include<iostream>
using namespace std;

class Base
{
  public: Base() { cout<<1; }
    Base(Base&b){ cout<<2; }
    ~Base() { cout<<3; } };
int f(int a,int b)
{
  if(b==0)
  {
    Base b; throw b;
  }
  else return a/b;
}
int main()
{
  try { cout<<f(9,0); }
  catch(Base b) { cout<<4; }
}
//VC下运行结果1223433
//CB下运行结果1232433

主要说一下在CB下的那种情况,编译器不一样那在这种创建临时对象的情况下优化的也就不一样。
主函数里try去调用函数f然后在f中b==0,创建一个对象b,调构造函数输出1,throw b,与return很相似,创建临时对象,调拷贝构造输出2,然后离开函数f,调析构函数,f里的b析构了输出3,到catch 把临时对象初始化形参b,调拷贝构造输出2,然后输出4,主函数结束,临时对象和形参b都析构输出两次2,即1232433。
小结:在对于返回和抛出一个对象的时候其实throw和return的相似度很高,内部的操作也十分相似。

return和throw某些特性相似的更多相关文章

  1. 错误处理之try、catch、finally中的return、throw执行顺序。

    今天遇到一个让人无语的代码块 try { bilSheetService.syncUser(bilWebseviceLog, userId, optType); }catch (Exception e ...

  2. java异常处理-finally中使用return和throw语句

    java异常语句中的finally块通常用来做资源释放操作,如关闭文件.关闭网络连接.关闭数据库连接等.正常情况下finally语句中不应该使用return语句也不应该抛出异常,以下讨论仅限于java ...

  3. java 異常抛出 throw 與 return

    package 異常;    public class TestException {      public TestException() {      }        boolean test ...

  4. node中的流程控制中,co,thunkify为什么return callback()可以做到流程控制?

    前言 我在学习generator ,yield ,co,thunkify的时候,有许多费解的地方,经过了许多的实践,也慢慢学会用,慢慢的理解,前一阵子有个其他项目的同事过来我们项目组学习node,发现 ...

  5. C++的优秀特性6:智能指针

    (转载请注明原创于潘多拉盒子) 智能指针(Smart Pointer)是C++非常重要的特性.考虑如下一段使用简单指针(Plain Pointer)的代码: A* a = new A(); B* b ...

  6. .NET面向对象特性之封装

    .NET面向对象特性之封装 面向对象的基本内容由:类.对象.属性.方法.字段构成. 面向对象的三大特性:继承.多态.封装. 关于面向对象的特性很多人都把目光转向了继承.多态和接口,却很少有人提及过封装 ...

  7. [干货来袭]C#7.0新特性(VS2017可用)

    前言 微软昨天发布了新的VS 2017 ..随之而来的还有很多很多东西... .NET新版本 ASP.NET新版本...等等..太多..实在没消化.. 分享一下其实2016年12月就已经公布了的C#7 ...

  8. 轻松学会ES6新特性之生成器

    生成器虽然是ES6最具魔性的新特性,但也是最难懂得的一节,笔者写了大量的实例来具体化这种抽象的概念,能够让人一看就懂,目的是希望别人不要重复或者减少笔者学习生成器的痛苦经历. 在说具体的ES6生成器之 ...

  9. C#7.0新特性

    前言 微软昨天发布了新的VS 2017 ..随之而来的还有很多很多东西... .NET新版本 ASP.NET新版本...等等..太多..实在没消化.. 分享一下其实2016年12月就已经公布了的C#7 ...

随机推荐

  1. Laravel-google-authenticator--Google验证码

    开发前的准备 安装Laravel 安装二维码生成器QrCode,没有安装也可以,接下来会安装 安装拓展 1.运行如下代码安装拓展包: composer require "earnp/lara ...

  2. GitLab CI/CD 进行持续集成

    简介 从 GitLab 8.0 开始,GitLab CI 就已经集成在 GitLab 中,我们只要在项目中添加一个 .gitlab-ci.yml 文件,然后添加一个 Runner,即可进行持续集成. ...

  3. 移动端添加横向滚动条&隐藏

    添加横向滚动条ul { display: flex; overflow-x: auto; overflow-y: hidden; white-space: nowrap; }隐藏滚动条,保留滚动效果 ...

  4. 【算法】CRF(条件随机场)

    CRF(条件随机场) 基本概念 场是什么 场就是一个联合概率分布.比如有3个变量,y1,y2,y3, 取值范围是{0,1}.联合概率分布就是{P(y2=0|y1=0,y3=0), P(y3=0|y1= ...

  5. AFNetWroking 3.0 GET&POST基本使用

    ``` - (void)requestWithUrl:(NSString *)url params:(NSDictionary *)params methodType:(NSString *)meth ...

  6. 2018年最新JAVA面试题总结之JavaWeb(2)

    转自于:https://zhuanlan.zhihu.com/p/39522575 1.tomcat的优化方式?回答:Tomcat的优化我准备从三方面来说: 第一部分: 内存优化Tomcat的默认内存 ...

  7. 2018-2019-2 网络对抗技术 20165323 Exp1 PC平台逆向破解

    实验目的 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回显任何用户输入的字符串. 该程序同时包含另一个代码片段,getShe ...

  8. TP-Shop安装步骤教程(Windows版)

    TP-Shop安装步骤教程(Windows版) PS:首次发文,请多指教! 一.安装要求 1.PHP5.4以上,MYsql5.5以上. 2.需要Phpcurl,gd库.php_mysqli,php_o ...

  9. youtube去广告

    https://www.digitbin.com/youtube-ads-block/ 1. OGYouTube | Mod AdBlocker YouTube OGYouTube App is a ...

  10. 类中被final修饰的成员变量需要初始化

    类中被final修饰的成员变量需要初始化,否则编译不通过,因为final修饰后不能再赋值,因此必须初始化.