c++构造函数的知识在各种c++教材上已有介绍,不过初学者往往不太注意观察和总结其中各种构造函数的特点和用法,故在此我根据自己的c++编程经验总结了一下c++中各种构造函数的特点,并附上例子,希望对初学者有所帮助。

c++类的构造函数详解

一、 构造函数是干什么的

 1 class Counter
 2 {
 3
 4 public:
 5          // 类Counter的构造函数
 6          // 特点:以类名作为函数名,无返回类型
 7          Counter()
 8          {
 9                 m_value = 0;
10          }
11
12 private:
13
14          // 数据成员
15          int m_value;
16 }

该类对象被创建时,编译系统对象分配内存空间,并自动调用该构造函数->由构造函数完成成员的初始化工作

eg:    Counter c1;
        编译系统为对象c1的每个数据成员(m_value)分配内存空间,并调用构造函数Counter( )自动地初始化对象c1的m_value值设置为0

故:

构造函数的作用:初始化对象的数据成员。

二、 构造函数的种类

 1 class Complex
 2 {
 3
 4 private :
 5         double    m_real;
 6         double    m_imag;
 7
 8 public:
 9
10         //    无参数构造函数
11         // 如果创建一个类你没有写任何构造函数,则系统会自动生成默认的无参构造函数,函数为空,什么都不做
12         // 只要你写了一个下面的某一种构造函数,系统就不会再自动生成这样一个默认的构造函数,如果希望有一个这样的无参构造函数,则需要自己显示地写出来
13         Complex(void)
14         {
15              m_real = 0.0;
16              m_imag = 0.0;
17         }
18
19         //    一般构造函数(也称重载构造函数)
20         // 一般构造函数可以有各种参数形式,一个类可以有多个一般构造函数,前提是参数的个数或者类型不同(基于c++的重载函数原理)
21         // 例如:你还可以写一个 Complex( int num)的构造函数出来
22         // 创建对象时根据传入的参数不同调用不同的构造函数
23         Complex(double real, double imag)
24         {
25              m_real = real;
26              m_imag = imag;
27          }
28
29         //    复制构造函数(也称为拷贝构造函数)
30         //    复制构造函数参数为类对象本身的引用,用于根据一个已存在的对象复制出一个新的该类的对象,一般在函数中会将已存在对象的数据成员的值复制一份到新创建的对象中
31         //    若没有显示的写复制构造函数,则系统会默认创建一个复制构造函数,但当类中有指针成员时,由系统默认创建该复制构造函数会存在风险,具体原因请查询 有关 “浅拷贝” 、“深拷贝”的文章论述
32         Complex(const Complex & c)
33         {
34                 // 将对象c中的数据成员值复制过来
35                 m_real = c.m_real;
36                 m_img    = c.m_img;
37         }
38
39         // 类型转换构造函数,根据一个指定的类型的对象创建一个本类的对象
40         // 例如:下面将根据一个double类型的对象创建了一个Complex对象
41         Complex::Complex(double r)
42         {
43                 m_real = r;
44                 m_imag = 0.0;
45         }
46
47         // 等号运算符重载
48         // 注意,这个类似复制构造函数,将=右边的本类对象的值复制给等号左边的对象,它不属于构造函数,等号左右两边的对象必须已经被创建
49         // 若没有显示的写=运算符重载,则系统也会创建一个默认的=运算符重载,只做一些基本的拷贝工作
50         Complex &operator=( const Complex &rhs )
51         {
52                 // 首先检测等号右边的是否就是左边的对象本,若是本对象本身,则直接返回
53                 if ( this == &rhs )
54                 {
55                         return *this;
56                 }
57
58                 // 复制等号右边的成员到左边的对象中
59                 this->m_real = rhs.m_real;
60                 this->m_imag = rhs.m_imag;
61
62                // 把等号左边的对象再次传出
63                // 目的是为了支持连等 eg:    a=b=c 系统首先运行 b=c
64                // 然后运行 a= ( b=c的返回值,这里应该是复制c值后的b对象)
65                 return *this;
66         }
67
68 };

下面使用上面定义的类对象来说明各个构造函数的用法:

 1 void main()
 2 {
 3         // 调用了无参构造函数,数据成员初值被赋为0.0
 4         Complex c1,c2;
 5
 6         // 调用一般构造函数,数据成员初值被赋为指定值
 7         Complex c3(1.0,2.5);
 8         // 也可以使用下面的形式
 9         Complex c3 = Complex(1.0,2.5);
10
11         //    把c3的数据成员的值赋值给c1
12         //    由于c1已经事先被创建,故此处不会调用任何构造函数
13         //    只会调用 = 号运算符重载函数
14         c1 = c3;
15
16         //    调用类型转换构造函数
17         //    系统首先调用类型转换构造函数,将5.2创建为一个本类的临时对象,然后调用等号运算符重载,将该临时对象赋值给c1
18         c2 = 5.2;
19
20        // 调用拷贝构造函数( 有下面两种调用方式)
21         Complex c5(c2);
22         Complex c4 = c2;  // 注意和 = 运算符重载区分,这里等号左边的对象不是事先已经创建,故需要调用拷贝构造函数,参数为c2
23
24
25
26 }

三、思考与测验

1. 仔细观察复制构造函数

      Complex(const Complex & c){// 将对象c中的数据成员值复制过来m_real = c.m_real;m_img = c.m_img;}    

为什么函数中可以直接访问对象c的私有成员?

2. 挑战题,了解引用与传值的区别

 1  Complex test1(const Complex& c)
 2   {
 3           return c;
 4   }
 5
 6   Complex test2(const Complex c)
 7   {
 8          return c;
 9    }
10
11    Complex test3()
12    {
13           static Complex c(1.0,5.0);
14           return c;
15    }
16
17   Complex& test4()
18   {
19          static Complex c(1.0,5.0);
20          return c;
21   }
22
23   void main()
24   {
25         Complex a,b;
26
27         // 下面函数执行过程中各会调用几次构造函数,调用的是什么构造函数?
28
29        test1(a);
30        test2(a);
31
32        b = test3();
33        b = test4();
34
35        test2(1.2);
36        // 下面这条语句会出错吗?
37        test1(1.2);     //test1( Complex(1.2 )) 呢?
38   }

四、附录(浅拷贝与深拷贝)

上面提到,如果没有自定义复制构造函数,则系统会创建默认的复制构造函数,但系统创建的默认复制构造函数只会执行“浅拷贝”,即将被拷贝对象的数据成员的值一一赋值给新创建的对象,若该类的数据成员中有指针成员,则会使得新的对象的指针所指向的地址与被拷贝对象的指针所指向的地址相同,delete该指针时则会导致两次重复delete而出错。下面是示例:

【浅拷贝与深拷贝】

 1 #include <iostream.h>
 2 #include <string.h>
 3 class Person
 4 {
 5 public :
 6
 7         // 构造函数
 8         Person(char * pN)
 9         {
10               cout << "一般构造函数被调用 !\n";
11               m_pName = new char[strlen(pN) + 1];
12               //在堆中开辟一个内存块存放pN所指的字符串
13               if(m_pName != NULL)
14               {
15                  //如果m_pName不是空指针,则把形参指针pN所指的字符串复制给它
16                    strcpy(m_pName ,pN);
17               }
18         }
19
20         // 系统创建的默认复制构造函数,只做位模式拷贝
21         Person(Person & p)
22         {
23                   //使两个字符串指针指向同一地址位置
24                  m_pName = p.m_pName;
25         }
26
27         ~Person( )
28         {
29                 delete m_pName;
30         }
31
32 private :
33
34         char * m_pName;
35 };
36
37 void main( )
38 {
39         Person man("lujun");
40         Person woman(man);
41
42         // 结果导致   man 和    woman 的指针都指向了同一个地址
43
44         // 函数结束析构时
45         // 同一个地址被delete两次
46 }
47
48
49 // 下面自己设计复制构造函数,实现“深拷贝”,即不让指针指向同一地址,而是重新申请一块内存给新的对象的指针数据成员
50 Person(Person & chs);
51 {
52          // 用运算符new为新对象的指针数据成员分配空间
53          m_pName=new char[strlen(p.m_pName)+ 1];
54
55          if(m_pName)
56          {
57                  // 复制内容
58                 strcpy(m_pName ,chs.m_pName);
59          }
60
61         // 则新创建的对象的m_pName与原对象chs的m_pName不再指向同一地址了
62 }

转载于:https://www.cnblogs.com/shercy/p/7063359.html

C++ 各种构造函数相关推荐

  1. 派生类参数初始化列表和基类构造函数顺序

    今天被问到了一个问题,随便回了一句,父类还没有构建,怎么能初始化父类的成员. 派生类构造函数的参数初始化列表,为什么不能初始化基类的成员? 例如下面的是不可以的 class Rectangle : p ...

  2. C++ 复制构造函数或者拷贝构造函数

    复制构造函数 是一种特殊的构造函数,它在创建对象时,是使用同一类中之前创建的对象来初始化新创建的对象. 复制构造函数通常用于: 通过使用另一个同类型的对象来初始化新创建的对象. 复制对象把它作为参数传 ...

  3. C++ 笔记(17)— 类和对象(构造函数、析构函数、拷贝构造函数)

    1. 构造函数 构造函数是一种特殊的函数(方法),在根据类创建对象时被调用.构造函数是一种随着对象创建而自动被调用的函数,它的主要用途是为对象作初始化. 构造函数的名称与类的名称是完全相同的,并且不会 ...

  4. php构造和析构方法,php5构造函数与析构函数实例

    自php5起,有了构造函数与析构函数. 这使得php更富有面向对象的魅力了. 在php4时,构造函数用的是与类同名的函数来进行构造这个动作. 例如: 复制代码 代码示例: /* * myclass.p ...

  5. java super快速生成_为什么当我使用编辑器代码生成器时,eclipse会在构造函数中自动添加一个java super()方法?...

    正如@Kon在他的评论中提到的, Java中的一个空构造函数包含对超类构造函数的隐式调用. 此外,没有明确调用super()的非空构造函数将在顶部具有隐式调用. 离开super()调用是唯一的时候,如 ...

  6. 【C++】何时需要自定义拷贝构造函数和赋值符

    先来说结论:当类中有指针类型成员变量的时候,一定要自定义拷贝构造和赋值运算符 原因:当我们在有指针类成员变量的时候,还是用默认拷贝构造函数(拷贝构造函数执行的时候会调用赋值符),默认赋值为浅拷贝,会导 ...

  7. 对象特性-----拷贝构造函数的调用

    C++中拷贝构造函数调用通常三种情况: ****使用一个已经创建完毕的对象来初始化一个新对象 ****值传递的方式给函数参数传值 ****以值方式返回局部对象 #include<iostream ...

  8. 对象特性-构造函数和析构函数

    基础概念: 构造函数:是一种特殊的方法.主要用来在创建对象时初始化对象,即为对象成员变量赋初始值,总与new运算符一起使用在创建对象的语句中.特别的一个类可以有多个构造函数 ,可根据其参数个数的不同或 ...

  9. C++:构造函数作用及用法

    PS:写在前面 就是构造函数的作用可以这样理解,如果没有构造函数就是类里边只是声明了成员变量,成员函数,还有最后的对象,这样你在对该对象进行初始化赋值时就比较麻烦就得先调用成员函数对成员变量赋值,成员 ...

  10. C++ 拷贝构造函数和重载赋值运算符的区别

    文章目录 拷贝构造函数 重载赋值运算符 赋值运算符和拷贝构造函数最大区别是赋值运算符没有新的对象生成,而拷贝构造函数会生成新的对象. 为了更加形象 准确得描述 赋值运算符和拷贝构造函数得区别,将详细通 ...

最新文章

  1. docker 离线安装 mysql_docker 离线安装
  2. simulink中使用memory模块实现变量的累加和(离散积分器)
  3. 小型数据中心规划和设计原则
  4. 纪念互联网第一本众筹电子书Mycat权威指南获得成功
  5. 一个「神奇」的Python库,99%的人都爱!
  6. linux下安装oracle sqlplus以及imp、exp工具
  7. Python 执行代码的两种方式
  8. Matlab常见问题处理——错误使用 xlswrite (line 219),调用错误,调度异常: 参数错误。
  9. win8 打开计算机配置,win8.1 更改电脑配置无法使用
  10. Unity5 Roll A Ball项目笔记
  11. 程序员正在抛弃 Facebook
  12. VLAN专题之三:VLAN的访问链接
  13. python 判断某个字符是否为中文
  14. 阿里电话面试经过与总结
  15. ffmpeg视频裁剪,切割,crop裁剪相关
  16. hdu4489 The King’s Ups and Downs
  17. nuc972外部中断1实验代码
  18. Linux 域名解析实验
  19. 2019年广东工业大学腾讯杯新生程序设计竞赛(同步赛)I-迷途的怪物
  20. 强网杯2019逆向 just re lebel:string2hex(string2decimal是atoi) / ida识别字符串有点小Bug? / rdtsc / 3DES(可PEiD插件识别)

热门文章

  1. mysql5.7 高可用_基于MySQL 5.7多源复制及Keepalived搭建三节点高可用架构
  2. java validation_java bean validation 参数验证
  3. 全麦吐司和普通吐司的区别_全麦面包和普通面包的区别
  4. 安装linux系统报softlock,soft lockup 解决思路
  5. 如何将一个集合里的对象进行计算再排序
  6. 一位老司机谈谈掏心窝子的话
  7. java null布局_Java Swing 绝对布局管理方法,null布局
  8. python文件编译_我算是白学Python了,现在才知道原来Python是可以编译的
  9. oracle行转列和列转行,oracle行转列和列转行(pivot 和 unpivot 函数,wm_concat函数 )...
  10. 什么是电子路径用于连接计算机主板上的芯片,计算机的组成部分及功能346.doc...