C++const常成员

const.png

const关键字在C语言中的应用

常类型是指使用类型修饰符const说明的类型,常类型的变量或对象的值是不能被修改的

const的普通用法

int const index = 10;   //const修饰的变量不能被修改,且必须初始化
const int index = 10;
初始化和赋值
/****初始化****/
int temp = 10;  //定义变量的同时初始化
/****赋值****/
int temp;
temp = 10;      //给变量赋值

这个在C语言的基本数据类型里面有说过

const修饰指针

常量指针

const修饰 *p

int const *p;   //const修饰 *p
const int* p;   //常量指针 不能通过指针修改p指向的内容(不能修改指向的内容)
指针常量

const 修饰 p

int temp =10;
int* const p = &temp;  //指针常量 不能修改p的指向 且必须初始化

const用于函数的形参

void fun(const int *p);

const用于形参时说明了形参在函数内部不会被改变

const关键字在C++中的应用

const修饰成员变量

class TEST
{
    const int m_tmp;  //常成员变量
public:
    TEST() :m_tmp(10)  //相当于 const int m_a = 10;
    {
        //m_a = 10; //赋值相当于修改值   常变量不能修改
    }
};

常成员变量必须使用初始化列表初始化

const修饰成员函数

类成员函数形参表之后,函数体之前加上const,这样的函数成为常函数

class TEST
{
    int m_money;  
public:
    TEST()  {
         m_money=1000;
    }
    void show() const
    {
        //常成员函数不能修改变量的值,下面是错误示范
        //cout << "资产:" << m_money++ << endl;
        cout << "资产:" << m_money << endl;   //常函数不能修改变量的值
    }
};
const修饰的是什么?

在C语言里面const是用来修饰变量的,那么在C++里面const是用来修饰什么,在C++中const实际修饰的是this指针,上面的代码在编译器看来是这个样子的

class TEST
{
    int m_money;
public:
    TEST(/*TEST* const this*/){
        this->m_money = 100;
    }
    void show(/*const TEST* const this*/) const //修饰this
    {
        cout << "资产:" << this->m_money << endl;
    }
};
const修饰成员函数构成重载?

是的没错,构成重载,原因是形参不同,构成重载

函数原型相同的成员函数,常版本和非常版本构成重载

class TEST
{
    int m_money;  
public:
    TEST()  {
         m_money=1000;
    }
    void show(/*TEST* const this*/) //重载
    {
        cout<< "资产:" << m_money++ << endl;
    }
    void show(/*const TEST* const this*/) const
    {
        cout << "const 资产:" << m_money << endl;
    }
};
const成员函数的调用问题

常成员函数不能调用非常成员函数,非常成员函数可以调用常成员函数,如果有重载版本的非常函数,优先调用非常函数

#include<iostream>
using namespace std;
class TEST
{
    int m_money;
public:
    TEST() {
        m_money = 1000;
    }
    /***非常成员函数可以调用常成员函数,如果有重载版本的非常函数,优先调用非常函数***/
    void show(/*TEST* const this*/)
    {
        cout << "资产:" << m_money++ << endl;
        fun();
        foo();
    }
    /***********常成员函数不能调用非常成员函数***********/
    void show(/*const TEST* const this*/) const
    {
        cout << "const 资产:" << m_money << endl;
        //fun(this);    //错误原因不能将“this”指针从“const TEST”转换为“TEST &”   
        foo();
    }
    void fun(/*TEST* const this*/)
    {
        cout << "fun" << endl;
    }
    void foo()  const
    {
        cout << "const:fun" << endl;
    }
};
int main()
{
    TEST test;
    test.show();
    return 0;
}
打印结果:
资产:1000
fun
const:fun
  1. 常成员函数不能调用非常成员函数的原因:

是因为this指针的修饰符不同,在const成员函数里面的this指针被const修饰,在const成员函数里面调用非const成员函数时,相当于将const修饰的this指针传给非const成员函数。

  1. 非常成员函数可以调用常成员函数的原因:

这个涉及到编译器的原理:提高权限不安全,缩小权限是安全的

这个权限怎么理解:

/***********非常成员函数***********/
void show()
{   
    //this可读写
    cout << "资产:" << m_money++ << endl;
    fun();  //调用非常成员函数 this可读写
    foo();  //调用常成员函数 this可读
}
/***********常成员函数***********/
void show() const
{
    //this可读
    cout << "const 资产:" << m_money << endl;
    //fun(this);    //调用非常成员函数 this可读写  
    foo();          //调用常成员函数 this可读
}
void fun()
{
    cout << "fun" << endl;
}
void foo()  const
{
    cout << "const:fun" << endl;
}

this可读写代表可以访问和修改this里面的成员变量的值

this可读代表可以访问this里面的成员变量的值

  1. 非常成员函数可以调用常成员函数,如果有重载版本的非常函数,优先调用非常函数

这个从权限来讲,编译器坚定不移的执行了不背锅精神,编译器认为缩小权限是安全的,扩大权限是危险的,编译器会优先选择最合适的

const修饰对象

被const修饰的对象 意味着对象中的成员数据不可以改内容

  • 常对象不能调用非常成员函数,只能调用常函数
  • 非常对象(普通对象)可以调用常成员函数如果有重载版本的非常函数,优先非常

原理和const修饰成员函数的原理是一样的,理解了const修饰成员函数 这一部分,那么const修饰对象也很容易理解

关键字 mutable

在C++里面有个在常成员函数中开VIP的存在,这就是mutable

mutable int m_money;//mutable修饰的变量可以在常成员函数中修改
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。