构造函数是C++、Java等面向对象编程语言不可绕过的话题,构造函数的作用就是初始化所创建对象的数据成员。不过与Java相比C++的构造函数显得更为复杂。因此,现在便来讲解一下C++的构造函数,以此进一步加深对构造函数的理解。

C++的构造函数分为无参构造函数和有参构造函数,且构造函数可以重载。当一个类中没有自定义的构造函数时,创建对象时便会自动调用系统默认的无参构造函数,不过这个构造函数的函数体为空,什么操作也不执行。不过一旦有自定义的构造函数时,便再也不会调用系统默认的构造函数。如果要使用无参构造函数,便需要自行定义。

数据成员为基本类型的类的讲解:

示例代码如下:

 #include <iostream>
using namespace std; class Test
{
public:
Test();
Test(int a, int b);
private:
int a;
int b;
}; Test::Test()
{
cout << "调用Test的无参构造函数" << endl;
a = ;
b = ;
} Test::Test(int a, int b)
{
cout << "调用Test的有参构造函数" << endl;
this->a = a;
this->b = b;
} int main()
{
//若想通过无参构造函数初始化对象,只能以此种方式声明对象
Test t1;
Test t2(, );
Test t3 = Test(, ); Test* t11 = new Test;
Test* t22 = new Test(, );
}

运行结果如下:

通过分析运行结果我们可以发现:

Test t1;这个语句所执行的操作是:在栈区为对象t1分配存储空间,并调用无参构造函数对t1数据成员进行初始化。

Test* t11 = new Test;这个语句所执行的操作是:在栈区为t11指针分配存储空间,在堆区为t11所指向的对象分配存储空间,并调用默认构造函数对对象的数据成员进行初始化。

数据成员为类类型的类的讲解:

示例代码:

 #include <iostream>
#include <vector> using namespace std; class Person
{
public:
Person();
Person(string name, int age);
void display();
private:
string name;
int age;
}; Person::Person()
{
cout << "调用 Person 无参构造函数" << endl;
name = "kkkk";
age = ;
} Person::Person(string name, int age)
{
cout << "调用 Person 有参构造函数" << endl;
this->name = name;
this->age = age;
} void Person::display()
{
cout << "Name: " << name << " Age: " << age << endl;
} class Test
{
public:
Test();
Test(string name, int age, int value);
void show();
private:
Person person;
int value;
}; Test::Test()
{
person = Person("wwww", );
person.display();
cout << "调用 Test 无参构造函数" << endl;
value = ;
} Test::Test(string name, int age, int value):person(name, age)
{
person.display();
cout << "调用 Test 第二个有参构造函数" << endl;
value = ;
} void Test::show()
{
person.display();
cout << "Value: " << value << endl;
} int main()
{
Test t1;
cout << "##################################" << endl;
Test t2("hhhh", , );
return ;
}

运行结果如下:

分析运行结果:

Test::Test()
{
    person = Person("wwww", 3333);
    person.display();
    cout << "调用 Test 无参构造函数" << endl;
    value = 100;
}

此构造函数实际是首先调用person的无参构造函数初始化person,然后再初始化本对象基本数据类型的数据成员。

而person = Person("wwww", 3333);语句是将Person("wwww", 3333)对象的数据成员复制给本person对象的数据成员,而且通过=复制本身也只是一种浅拷贝。相当于person对象被进行了两次初始化

Test::Test(string name, int age, int value):person(name, age)
{
    person.display();
    cout << "调用 Test 第二个有参构造函数" << endl;
    value = 100;
}

这个Test函数与上一个Test并无本质区别,只是这个Test函数是显式调用person对象的有参构造函数。

由此可以得出以下结论:

有类A,A有类类型数据成员B b,那么A的构造函数只能是以下形式:

如果B有无参构造函数:

A::A()

{

  ......

}

A::A(...)

{

  ......

}

如果B有有参构造函数:

A::A():b(...)

{

  .....

}

A::A(...):b(...)

{

  ......

}

继承时构造函数的讲解:

示例代码:

 #include <iostream>

 using namespace std;

 class A
{
public:
A();
A(int a);
private:
int a;
}; A::A()
{
cout << "调用A无参构造函数" << endl;
a = ;
} A::A(int a)
{
cout << "调用A有参构造函数" << endl;
this->a = a;
} class B: public A
{
public:
B();
B(int a, int b);
private:
int b;
}; B::B()
{
cout << "调用B无参构造函数" << endl;
b = ;
} B::B(int a, int b):A(a)
{
cout << "调用B有参构造函数" << endl;
this->b = b;
} int main()
{
B b1;
cout << "##################################" << endl;
B b2(, );
return ;
}

运行结果如下:

分析运行结果:

B::B()
{
    cout << "调用B无参构造函数" << endl;
    b = 20;
}

此构造函数调用了A类的无参构造函数,且首先执行A类的构造函数,后执行B类的构造函数

B::B(int a, int b):A(a)
{
    cout << "调用B有参构造函数" << endl;
    this->b = b;
}

此构造函数与上一构造函数并无本质区别,只是显示调用A类的有参构造函数

由此可以得出以下结论:

现有类A,类B,且B继承A

如果A有无参构造函数,那么B的构造函数可以这么写:

B::B()

{

  ......

}

B::B(...)

{

  ......

}

如果A有有参构造函数,那么B的构造函数可以这么写:

B::B():A(...)

{

  ......

}

B::B(...):A(...)

{

  ......

}

讲解完毕,若有不足之处,敬请指出!!!

C++ 构造函数讲解的更多相关文章

  1. js中自定义构造函数讲解

    什么是构造函数? 构造函数其实就是一个函数,只是用途和普通函数,不太一样, 构造函数一般用于初始化对象 <script> function Person(){ this.name=&quo ...

  2. 慕课网:C++ & 数据结构

    课程链接:james_yuan的课程 这部分太枯燥了,如果教材难度稍大,根本就学不下去,所以就先看看通俗的视频吧. 课程目录 1.C++远征之起航篇 - C++亮点尽在其中 2.C++远征之离港篇 - ...

  3. C++视频课程小结(3)

    C++远征之封装篇(上) 章节介绍: 每章小结: 第一章:课程介绍. 按照惯例是章节的总介绍,内容明显多了很多(为了做作业我还要赶进度的说),主要说了:类和对象是本章的主角,然后还有很多配角,像数据成 ...

  4. Java源码分析:关于 HashMap 1.8 的重大更新(转载)

    http://blog.csdn.net/carson_ho/article/details/79373134 前言 HashMap 在 Java 和 Android 开发中非常常见 而HashMap ...

  5. 6-C++远征之封装篇[上]-学习笔记

    C++远征之封装篇(上) 课程简介 类(抽象概念),对象(真实具体) 配角: 数据成员和成员函数(构成了精彩而完整的类) 构造函数 & 析构函数(描述了对象的生生死死) 对象复制和对象赋值 ( ...

  6. javascript面向对象基础讲解(工厂模式、构造函数模式、原型模式、混合模式、动态原型模式)

    面向对象可以把程序中的关键模块都视为对象,而模块拥有属性及方法.这样我们如果把一些属性及方法封装起来,日后使用将非常方便,也可以避免繁琐重复的工作.接下来将为大家讲解在JS中面向对象的实现.   工厂 ...

  7. JavaScript继承基础讲解,原型链、借用构造函数、混合模式、原型式继承、寄生式继承、寄生组合式继承

    说好的讲解JavaScript继承,可是迟迟到现在讲解.废话不多说,直接进入正题. 既然你想了解继承,证明你对JavaScript面向对象已经有一定的了解,如还有什么不理解的可以参考<面向对象J ...

  8. 面向对象JS基础讲解,工厂模式、构造函数模式、原型模式、混合模式、动态原型模式

    什么是面向对象?面向对象是一种思想!(废话). 面向对象可以把程序中的关键模块都视为对象,而模块拥有属性及方法.这样我们如果把一些属性及方法封装起来,日后使用将非常方便,也可以避免繁琐重复的工作.接下 ...

  9. JavaScript 中实现继承的方式(列举3种在前一章,我们曾经讲解过创建类的最好方式是用构造函数定义属性,用原型定义方法。)

    第一种:对象冒充 function ClassA(sColor) { this.color = sColor; this.sayColor = function () { alert(this.col ...

随机推荐

  1. Device Pixel Ratio & Media Queries

    一些重要的名词解释: CSS pixels(CSS 像素):详见http://www.w3.org/TR/css3-values/#reference-pixe CSS声明的像素值,可随着放大缩小而放 ...

  2. Linux内核高端内存 转

        Linux内核地址映射模型x86 CPU采用了段页式地址映射模型.进程代码中的地址为逻辑地址,经过段页式地址映射后,才真正访问物理内存. 段页式机制如下图.   Linux内核地址空间划分 通 ...

  3. LMAX Disruptor – High Performance, Low Latency and Simple Too 转载

    原文地址:http://www.symphonious.net/2011/07/11/lmax-disruptor-high-performance-low-latency-and-simple-to ...

  4. 10 ways to be a faster code reviewer--reference

    reference:http://blog.codacy.com/top-10-faster-code-reviews/ This is a blog post of our Code Reading ...

  5. Linux服务器集群系统(二)--转

    引用地址:http://www.linuxvirtualserver.org/zh/lvs2.html LVS集群的体系结构 章文嵩 (wensong@linux-vs.org) 2002 年 4 月 ...

  6. Enterprise Library深入解析与灵活应用(2): 通过SqlDependency实现Cache和Database的同步

    对于一个真正的企业级的应用来说,Caching肯定是一个不得不考虑的因素,合理.有效地利用Caching对于增强应用的Performance(减少对基于Persistent storage的IO操作) ...

  7. C#之结构化异常处理

    1.异常处理4要素 一个表示异常详细信息的类类型: 一个向调用者引发异常类实例的成员: 调用者的一段调用异常成员的代码块: 调用者的一段处理将要发生异常的代码块:

  8. 基于 SOA 的组件化业务基础平台

    业务基础平台是业务逻辑应用和基础架构平台之间的一个中间层,解决 “应用软件的业务描述和操作系统平台.软件基础架构平台之间的交互与管理问题”.操作系统平台解决了“应用软件系统与硬件之间的交互与管理问题” ...

  9. VMware系统运维(十四)部署虚拟化桌面 Horzion View Manager 5.2 配置许可

    1.通过网页访问https://conntionserver.testad.local/admin,打开如下图所示页面:输入用户名密码 2.进来以后配置View 许可,点击"编辑许可证&qu ...

  10. 解决IE下z-index的bug

    IE下z-index有很奇葩的BUG,就是z-index级别不生效.解决方案只有在 该元素的父级元素添加 posotion和z-index,就能识别了. element{position:relati ...