20-初始化列表的使用

注:博客中内容主要来自《狄泰软件学院》,博客仅当私人笔记使用。

测试环境:Ubuntu 10.10

GCC版本:4.4.5

 

一、问题

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

       类中可以宏定义const成员变量,只能通过成员列表进行初始化。

 

二、小实验

1)下面的类定义是否合法?

      - 如果合法ci的值是什么存在在哪里

class Test
{
private:
    const int ci;
public:
    int getCI() {return ci;}
};

 

编程实验
类中的const成员
20-1.cpp
#include <stdio.h>

class Test
{
private:
    const int ci;
public:
    Test() : ci(10)
    {

    }

    int getCI()
    {
        return ci;
    }
};

int main()
{
    Test t;

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

    return 0;
}

操作:

1)g++ 20-1.cpp -o 20-1.out编译正确,打印结果:

t.ci = 10

2)测试:不在构造函数的初始化列表初始化const修饰的变量ci,代码如下:

#include <stdio.h>

class Test
{
private:
    const int ci;
public:
    Test()
    {
        ci = 10;    //编译器会报错,提示ci是只读变量
    }

    int getCI()
    {
        return ci;
    }
};

int main()
{
    Test t;

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

    return 0;
}

操作:g++ 20-1.cpp -o 20-1.out编译错误:

20-1.cpp:8:2 error: uninitialized member 'Test::ci' with 'const' type 'const int' 
     Test()
20-1.cpp:10:6: error: assignment of read-only member 'Test::ci'
     ci = 10;

 

三、类成员的初始化

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

2)语法规则

Class Name::Class Name() : m1(v1), m2(v1,v2), m3(v3)    //初始化列表对成员变量进行初始化
{
    //some other initialize operation
}

3)注意事项

      - 成员的初始化顺序与成员的声明顺序相同

      - 成员的初始化顺序与初始化列表中的位置无关

      - 初始化列表先于构造函数的函数体

编程试验
初始化列表的使用
20-2.cpp

#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 m3;    //声明成员
    Value m1;    //声明成员
public:
    Test() : m1(1), m2(2), m3(3)
    {
        printf("Test::Test()\n");
    }
};

int main()
{
    Test t;

    return 0;
}

操作:

1)编译:g++ 20-2.cpp -o 20-2.out,打印结果:

i = 2
i = 3
i = 1
Test::Test()

分析:

       打印结果证明了先执行初始化列表,在执行构造函数。成员变量声明顺序,决定了变量在初始化列表声明顺序。

 

四、类中的const成员

1)类中的const成员会被分配空间的(和类空间位置一致)

2)类中的const成员的本质是只读变量

3)类中的const成员只能在初始化列表中指定初始值

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

编程实验
只读成员变量
20-3.cpp

#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;
    }

    void setCI(int v)
    {
        int* p = const_cast<int*>(&ci);	//证明ci变量是只读的,没有进入符号表
        *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;
}

操作:

1) g++ 20-3.cpp -o 20-3.out编译正确,运行结果:

i = 2
i = 3
i = 1
Test::Test()
t.ci = 100
t.ci = 10

分析:

        const修饰的成员变量,并不是真正意义的常量,可以通过指针进行修改。

 

五、小插曲

1)初始化与赋值不同

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

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

初始化和赋值的不同
int main()
{
    int i = 0;	//初始化i的值

    i = 0;	//赋值
	
    return 0;
}

 

小结

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

2)初始化列表先于构造函数体执行

3)类中可以定义const成员变量

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

5)const成员变量为只读变量

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值