构造函数是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. Java IO和Java NIO在文件拷贝上的性能差异分析

    1.  在JAVA传统的IO系统中,读取磁盘文件数据的过程如下: 以FileInputStream类为例,该类有一个read(byte b[])方法,byte b[]是我们要存储读取到用户空间的缓冲区 ...

  2. php 换行 空格分割处理

    <?php function parse_specification($specification){ $rt=array(); $lines=array_filter(preg_split(& ...

  3. spring security源码分析之core包

    Spring Security是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架.它提供了一组可以在Spring应用上下文中配置的Bean,充分利用了Spring I ...

  4. js的Touch事件

    js的touch事件,一般用于移动端的触屏滑动$(function(){document.addEventListener("touchmove", _touch, false); ...

  5. Java基础知识强化之多线程笔记05:Java中继承thread类 与 实现Runnable接口的区别

    1. Java中线程的创建有两种方式:  (1)通过继承Thread类,重写Thread的run()方法,将线程运行的逻辑放在其中. (2)通过实现Runnable接口,实例化Thread类. 2. ...

  6. 【Java/Android性能优化1】Android性能调优

    本文参考:http://www.trinea.cn/android/android-performance-demo/ 本文主要分享自己在appstore项目中的性能调优点,包括同步改异步.缓存.La ...

  7. 放飞App:移动产品经理实战指南

    <放飞App:移动产品经理实战指南> 基本信息 原书名:App savvy:rurning ideas into iPhone and iPad Apps customers really ...

  8. VMware系统运维(八)vCenter Server安装

    1.终于开始安装vCenter Server了,需要配置数据源哦! 2.下一步 3.接受协议,下一步 4.输入许可密钥,也可以后面再输入,下一步 5.选择数据源,即我们前面配置的系统DSN,下一步 6 ...

  9. poj 3017 单调队列优化动态规划

    思路:dp[i]=min{dp[j]+max(num[j+1]...num[i])},其中sum[i]-sum[j]<=m. 那么我们需要用单调队列维护j到i的最大值. #include< ...

  10. Atmega8型号细分区别

    TMEGA8L- 8PU.ATMEGA8L-8PI.ATMEGA8L-8PC.ATMEGA8-16PU.ATMEGA8-16PI.ATMEGA8-16PC. ATMEGA8L-8AU.ATMEGA8L ...