首先回忆下,以前学的const

单独使用const修饰变量时,是定义的常量,比如:const int i=1;

使用volatile const修饰变量时,定义的是只读变量

使用const & 修饰变量时,定义的是只读变量

在类中是否可以定义const成员?

直接来写代码:

#include <stdio.h>
class Test
{
private:
const int ci;
public:
// Test()
// {
// ci=10;
// }
int getCI()
{
return ci;
}
};
int main()
{
Test t;
printf("%d\n",t.getCI());
return ;
}

编译出错:

test.cpp: In function ‘int main()’:

test.cpp:: error: structure ‘t’ with uninitialized const members

从编译信息看出,由于结构体t的const成员没有初始化,所以执行printf()出错.

接下来取消上面示例的屏蔽,使用上章学习的构造函数来初始化const

编译还是出错:

test.cpp: In constructor ‘Test::Test()’:

test.cpp:: error: uninitialized member ‘Test::ci’ with ‘const’ type ‘const int’

test.cpp:: error: assignment of read-only data-member ‘Test::ci’

从编译信息看出, Test::Test()构造函数里,不能直接初始化const变量.

所以,在C++中,便引入了构造函数初始化列表(除了可以给成员变量初始化,还可以对const成员初始化

初始化列表位于构造函数名右侧,以一个冒号开始,接着便是需要初始化的变量,以逗号隔开,例如:

class Example
{
private:
int i;
float j;
const int ci;
int *p;
public:
Test(): j(1.5),i(),ci() //初始化i=2,j=1.5,ci=10
{
p=new int;
*p=;
}
};

注意:

-列表成员的初始化顺序只与成员的声明顺序相同,与初始化列表的位置无关

比如上个示例,初始化列表初始化的顺序为: i=2,j=1.5,ci=10

-调用构造函数初始化时,会先执行初始化列表,再执行构造函数里的内容.

那class类里的const成员是常量还是只读变量?

参考以下示例:

#include <stdio.h>

class Test
{
private:
const int ci;
public:
Test():ci()
{ }
int getCI()
{
return ci;
}
void setCI(int val)
{
int *p=const_cast<int *>(&ci);
*p=val;
}
}; int main()
{
Test t;
t.setCI();
printf("%d\n",t.getCI());
return ;
}

编译运行:

   

所以class类里的const成员, 定义的是只读变量

对象的构造顺序

C++中的类可以定义多个对象,那么对象构造的顺序又是怎么样的?

对于局部对象(栈)

-程序执行到对象的定义语句时,便进行构造

对于通过new创建的对象(堆)

-和局部对象一样,程序执行到new语句时,便进行构造

对于全局对象(静态存储区)

-对象的构造顺序是不确定的,所以要尽量避免多个全局对象之间的相互依赖.

对象的销毁-析构函数

之前我们学习过创建对象时,有构造函数进行初始化.

同样的,对象被销毁前也应该要有一些清理工作,所以,C++中引入了一个特殊的清理函数-析构函数

  • 析构函数的功能与构造函数相反,在对象被摧毁时自动调用
  • 析构函数没有参数,也没有返回值类型声明

定义为: ~class_name(),例如:

class Test{
public:
Test(){ } //构造函数
~Test(){ } //析构函数
};

注意:

  • 在类里,当定义了析构函数,编译器就不会提供默认的构造函数了,所以还要自己定义一个构造函数。
  • 使用new创建的对象变量,在不使用时,需要使用delete,才能调用析构函数

参考以下示例:

#include <stdio.h>

class Test
{
int val; public:
Test(int i)
{
val=i;
printf("Test() val=%d\n",val);
}
~Test()
{
printf("~Test() val=%d\n",val);
}
}; int main()
{
Test t1();
Test* t2 = new Test(); // delete t2;
return ;
}

编译运行:

Test()
Test()
~Test()

从打印结果可以看出,t2的析构函数没有打印,所以只打印了:~Test(1)

取消屏蔽后再次运行:

Test()
Test()
~Test()
~Test()

总结:

当类中有成员需要内存申请,文件打开,链接数据库等时,则需要定义析构函数,进行回收资源

(和拷贝构造函数类似)

10.C++-构造函数初始化列表、类const成员、对象构造顺序、析构函数的更多相关文章

  1. <四>构造函数初始化列表

    示例代码1 点击查看代码 class CDate{ public: CDate(int _year,int _month, int _day){ this->year=_year; this-& ...

  2. 从零开始学C++之构造函数与析构函数(二):初始化列表(const和引用成员)、拷贝构造函数

    一.构造函数初始化列表 推荐在构造函数初始化列表中进行初始化 构造函数的执行分为两个阶段 初始化段 普通计算段 (一).对象成员及其初始化  C++ Code  1 2 3 4 5 6 7 8 9 1 ...

  3. 初始化列表(const和引用成员)、拷贝构造函数

    一.构造函数初始化列表 推荐在构造函数初始化列表中进行初始化 构造函数的执行分为两个阶段 初始化段 普通计算段 (一).对象成员及其初始化  C++ Code  1 2 3 4 5 6 7 8 9 1 ...

  4. const成员或者引用成员必须使用构造函数初始化列表的方式

    #include<iostream.h> class A { const int a; int b; }; void main() { A obja; }编译出现如下错误:error C2 ...

  5. C++类构造函数初始化列表

    C++类构造函数初始化列表 构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式.例如: class CExample {public:     ...

  6. C++类构造函数初始化列表(转)

    构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式.例如: { public:     int a;     float b;     //构 ...

  7. C++中为什么构造函数初始化列表

    已经有个构造函数负责初始化,为什么还需要构造函数初始化表呢? 在以下三种情况下需要使用初始化成员列表:一,需要初始化的数据成员是对象的情况:二,需要初始化const修饰的类成员:三,需要初始化引用成员 ...

  8. C++构造函数初始化列表与构造函数中的赋值的区别

    C++类中成员变量的初始化有两种方式:构造函数初始化列表和构造函数体内赋值. 一.内部数据类型(char,int……指针等) class Animal { public: Animal(int wei ...

  9. C++:用成员初始化列表对数据成员初始化

    1.在声明类时,对数据成员的初始化工作一般在构造函数中用赋值语句进行. 例如: class Complex{ private: double real; double imag; public: Co ...

随机推荐

  1. ASP.NET MVC下使用AngularJs语言(四):$window.alert

    判断文本框是否有填写,没有填写使用angularjs的$window.alert来提示用户. 创建一个ASP.NET MVC控制器: 接下来是准备一个angularjs的控制器: pilotApp.c ...

  2. 简单的异步函数async/await例子

    function resolveAfter2Seconds(x){ return new Promise(resolve => { setTimeout(() => { resolve(x ...

  3. [CocoaPods]客户端加载第三方库

    请先阅读另一篇博文铺垫知识基础:[CocoaPods]终端方式集成第三方库 客户端的Github地址:CocoaPods-app 点击下载客户端: [CocoaPods客户端] 安装下载的文件.软件界 ...

  4. CUDA driver version is insufficient for CUDA runtime version 解决

    配置ubuntu17.1+CUDA9.2的caffe环境,CUDA sample编译完成,执行到./deviceQuery时报错:CUDA driver version is insufficient ...

  5. JavaScript 交换数组元素位置的几种方式

    前言 交换数组元素位置是开发项目中经常用到的场景,总结下用过的几种方式. 第三方变量 最基础的方式,创建一个变量作为中转. let temp = array[index1]; array[index1 ...

  6. Go 新起点

    因项目需求 又得开始啃Go了,虽然比计划早了点,撸起袖子开始干吧~

  7. 简单标签SimpleTag

    想要开发自定义标签,大多数情况下都要重写doStartTag(),doAfterBody()和doEndTag()方法,并且还要知道SKIP_BODY,EVAL_BODY等等的变量代表着什么,在什么方 ...

  8. [每天解决一问题系列 - 0005] WiX Burn 如何校验chained package的合法性

    问题描述: 项目中使用Wix burn打包,内部包含了多个MSI.有时候会遇到如下错误 Error 0x80091007: Failed to verify hash of payload: Setu ...

  9. Django--models--多表操作

    一 创建模型 实例:我们来假定下面这些概念,字段和关系 作者模型:一个作者有姓名和年龄. 作者详细模型:把作者的详情放到详情表,包含生日,手机号,家庭住址等信息.作者详情模型和作者模型之间是一对一的关 ...

  10. Java高并发之设计模式

    本文主要讲解几种常见并行模式, 具体目录结构如下图. 单例 单例是最常见的一种设计模式, 一般用于全局对象管理, 比如xml配置读写之类的. 一般分为懒汉式, 饿汉式. 懒汉式: 方法上加synchr ...