十八、初始化列表的使用

1、类中定义const成员,及其初始化

#include <stdio.h>
class Test
{
private:
    const int ci;   // 编译通过,类中可以用const来修饰
public:
    int getCI()
    {
        return ci;
    }
};

int main()
{
    Test t; // 定义t对象的时候,报错说ci没有初始化

    printf("t.ci = %d\n", t.getCI());

    return 0;
}

如何初始化类里面的const成员?

首先考虑构造函数初始化

class Test
{
private:
    const int ci;   // 得到的是只读变量,不能作为左值
    // 初始化必须在这里进行
public:
    Test()
    {
        ci = 10;    // ci是只读成员变量,不能修改,报错
    }
    int getCI()
    {
        return ci;
    }
}

于是出现了初始化列表

C++中提供了初始化列表对成员变量进行初始化

语法规则:

// 构造函数
// 初始化列表在构造函数的地方使用
// 在构造函数的定义之后,函数体之前,加 :号,初始化列表对成员变量进行初始化
// 用v1对m1进行初始化,用v1,v2对m2进行初始化,用v3对m3进行初始化
ClassName::ClassName() : m1(v1), m2(v1,v2), m3(v3)
{}

通过初始化列表对ci 进行初始化

class Test
{
private:
    const int ci;
public:
    Test():ci(10)   // 编译通过,用初始化列表的方式初始化了const只读成员变量,初始化之后,还是可以通过指针的方式修改这个值,只是ci不能做左值
    {
        // ci = 10;
    }
    int getCI()
    {
        return ci;
    }
};

类成员的初始化, 注意事项:

  • 成员的初始化顺序与成员的申明顺序相同
  • 成员的初始化顺序与初始化列表中的位置无关
  • 初始化列表先于构造函数的函数体执行
// 初始化顺序和初始化列表无关,和声明顺序相同
ClassName::ClassName() : m1(v1), m2(v1,v2), m3(v3)
{}

构造函数是在对象创建完成之后,再调用构造函数进行成员变量的初始化,其实是一个成员变量的赋值函数,并非真正意义上的初始化

初始化列表则是在对象创建的同时就对对象进行了初始化,顺序先于构造函数函数体

区别类似于变量的初始化和赋值

int a = 1;  // 类似于初始化列表

int a;
a = 1;      // 类似于构造函数
#include <stdio.h>

class Value
{
private:
    int mi;
public:
    Value(int i)
    {
        printf("i = %d\n", i);
        mi = i;
    }
    int getI()
    {
        return mi;
    }
};

class Test
{
private:
    Value m2;   // Value 类创建的对象,带了1个参数
    // Value m2(2); // err
    Value m3;
    Value m1;
public:
    Test() : m1(1), m2(2), m3(3)    // 使用初始化列表来进行初始化
    {
        printf("Test::Test()\n");
    }
};

int main()
{
    Test t;     // 打印列表是 2 3 1 , 顺序按照成员变量声明顺序

    return 0;
}

类中的const成员

  • 类中的const成员会被分配空间
  • 类中的const成员本质是只读变量
  • 类中的const成员只能在初始化列表中制定初始值

编译器无法直接得到const成员的初始值,因此无法进入符号表成为真正意义上的常量。

const成员分配的空间的位置和当前对象一样,对象在栈区,const成员就在栈区

#include <stdio.h>

class Value
{
private:
    int mi;
public:
    Value(int i)
    {
        printf("i = %d\n", i);
        mi = i;
    }
    int getI()
    {
        return mi;
    }
};

class Test
{
private:
    const int ci;
    Value m2;
    Value m3;
    Value m1;
public:
    Test() : m1(1), m2(2), m3(3), ci(100)
    {
        printf("Test::Test()\n");
    }
    int getCI()
    {
        return ci;
    }
    int setCI(int v)
    {
        int *p = const_cast<int *>(&ci);    // 用指针改变const成员变量的值
        *p = v;
    }
};

int main()
{
    Test t;

    printf("t.ci = %d\n", t.getCI());

    t.setCI(10);

    printf("t.ci = %d\n", t.getCI());

    return 0;
}

初始化与赋值的区别:

初始化:对正在创建的对象进行初始值设置

赋值:对已经存在的对象进行值设置

2、小结

类中可以使用初始化里列表对成员进行初始化

初始化列表先于构造函数体进行

类中可以定义const成员变量

const成员变量必须在初始化列表中指定初值

const成员变量为只读变量

原文地址:https://www.cnblogs.com/chenke1731/p/9703041.html

时间: 2024-11-09 03:19:33

十八、初始化列表的使用的相关文章

C++(二十八) — 构造函数的初始化列表

1.解决的问题: 在 B 类中,组合了一个 A 类对象,其中A类设计了构造函数.由于构造函数的调用规则,设计了构造函数就必须调用,但在定义B类时没有机会初始化A,因此采用构造函数的初始化列表来解决.  2.构造和析构调用顺序 先执行被组合对象的构造函数,如果组合对象有多个,则按照定义顺序,而不是按照初始化列表顺序.   析构函数,和构造函数的调用顺序相反. 举例如下: class A { public: A(int a_) { a = a_; cout << "A的构造函数,a: &

Bootstrap &lt;基础二十八&gt;列表组

列表组.列表组件用于以列表形式呈现复杂的和自定义的内容.创建一个基本的列表组的步骤如下: 向元素 <ul> 添加 class .list-group. 向 <li> 添加 class .list-group-item. 下面的实例演示了这点: <!DOCTYPE html> <html> <head> <title>Bootstrap 实例 - 基本的列表组</title> <link href="/boo

初始化列表的使用(十五)

我们之前在 C 语言中可以定义 const 成员,那么我们是否可以在类中定义 const 成员呢?我们来看看下面代码中的类定义是否合法呢?如果合法,ci 的值是什么,存储在哪里呢? #include <stdio.h> class Test { private:     const int ci; public:     int getCI()     {         return ci;     } }; int main() {     Test t;          printf(

NeHe OpenGL教程 第三十八课:资源文件

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第三十八课:资源文件 从资源文件中载入图像: 如何把图像数据保存到*.exe程序中,使用Windows的资源文件吧,它既简单又实用. 欢迎来到NeHe教程第38课.离上节课的写作已经有些时日了,加上写了一整天的code,也许笔头已经

十八、Android引导界面

一.所需素材 很有必要整理一下,里面附带友盟的社会化分享组件,我就不去掉了. 二.代码 import com.umeng.update.UmengUpdateAgent; import android.app.Activity; import android.content.Intent; import android.content.SharedPreferences; import android.os.Bundle; import android.os.Handler; import an

NeHe OpenGL教程 第二十八课:贝塞尔曲面

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第二十八课:贝塞尔曲面 贝塞尔曲面: 这是一课关于数学运算的,没有别的内容了.来,有信心就看看它吧. 贝塞尔曲面 作者: David Nikdel ( [email protected] ) 这篇教程旨在介绍贝塞尔曲面,希望有比我更

18、蛤蟆的数据结构笔记之十八链表实现稀疏矩阵

18.蛤蟆的数据结构笔记之十八链表实现稀疏矩阵 本篇名言:"必须如蜜蜂一样,采过许多花,才能酿出蜜来." 上篇中实现了栈在多项式实现中的例子,再来看下稀疏矩阵通过链表方式实现. 关键字:十字链表存储 欢迎转载,转载请标明出处: 1.  十字链表存储 十字链表(OrthogonalList)是有向图的另一种链式存储结构.该结构可以看成是将有向图的邻接表和逆邻接表结合起来得到的.用十字链表来存储有向图,可以达到高效的存取效果.同时,代码的可读性也会得到提升. 为便于理解后续代码,从网上摘了

会员卡管理系统技术解析(十八)Timer定时监听

会员卡管理系统技术解析(十八)Timer定时监听 在web应用中,有时候客户需要一些定时程序,不需要客户自己去操作,而是由应用程序自行触发(代理)执行某些操作.这个时候监听与定时器的配合使用就基本可以实现这个需求了.网上很多代码并不完整,水平未到,看得太吃力了.下面以系统每天00:00:00进行的当日最大单据数维护进行实例解析. 首先,建立相关的包河类,如下图: 图1 然后,在"web.xml"中对定时器和监听类注册,代码如下: <servlet> <servlet-

企业搜索引擎开发之连接器connector(二十八)

通常一个SnapshotRepository仓库对象对应一个DocumentSnapshotRepositoryMonitor监视器对象,同时也对应一个快照存储器对象,它们的关联是通过监视器管理对象DocumentSnapshotRepositoryMonitorManagerImpl实现的 DocumentSnapshotRepositoryMonitorManagerImpl类要实现那些行为,先查看其实现接口DocumentSnapshotRepositoryMonitorManager定义