C++ const成员函数

预备知识

1、代码转换分析技巧

在早期某些编译器会将C++代码翻译为C代码,然后使用C编译器生成可执行文件。其中翻译的一个转化就是:将this指针显式添加到成员函数的第一个参数位置上,因为C中没有OOP的支持,并在成员函数调用时,自动将对象的地址传递给参数this。

这个过程用如下代码解释:

#include<iostream>
#include<cstdio>
using namespace std;

class Dog
{
public:
    Dog(unsigned  en = 0) :energe(en){}  

    void bark() const
    {
        for (size_t i = 0; i < energe; i++)
            cout << "wang wang!\n";
    }

    void feed(unsigned add)
    {
        energe += add;
    }

private:
    unsigned  energe;
};

int main()
{

    Dog dog;

    dog.feed(2);
    dog.bark();    

    return 0;
}

翻译转换

//原型转换
void feed(const Dog*this);
void feed(Dog*this,unsigned add);

//调用的转换
dog.feed(2);   ----->  feed(&dog , 2);
dog.bark();    ----->  bark(&dog);

现在的C++编译器可能不是这样的工作的,但是,如果你使用这个转换方式去应用到实际的编程的代码分析中,很多代码和语法特性就会迎刃而解。

2、顶层const,底层const

顶层const:指针变量本身是常量。(顶层const不适合引用,因为引用天生就是一个常量,始终引用一个对象,直到消亡)

如   int* const p = &a;

底层const:变量指向(或者引用)的对象被视为常量。(注意这里的用词:被视为,因为对象不一定就是常量,也可能是底层const指针、引用的一厢情愿)

如    const int*p ;    int const *p;    这二者写法等价

const int& r;

一个非底层const指针,它对它指向的对象有 读、写的权利,因此,为了确保数据安全,它仅能指向非常量对象。

int a=1;

const int b = 2;

int*p1 = &a;  //OK

p1  = &b;      //Error不允许

一个底层const指针,他对它指向的对象仅有读的权利。因此,它可以指向常量和非常量。

int a=1;

const int b = 2;

const int*p1 = &a;  //OK

p1 = &b ;           //OK

3、函数的参数是否为 底层const 可以作为重载的依据

举例说,下面2个函数可以重载。

void foo(const int*p);     //A
void foo(int*p);           //B

编译器在重载解析时,根据传递的参数来判断调用哪一个版本。

当仅存在版本A时,若传递的是非常量int的指针,或常量int的指针,都可以成功调用。

当仅存在版本B时,若传递的是非常量int的指针,可以成功调用,但不允许传递常量int的指针。

当A、B 都存在,重载时,若传递的是非常量int的指针,则优先使用版本B,因为这是最匹配的。

若传递的是常量int的指针, 则使用版本A。

另外,参数本身是否是const 不作为重载的依据,下面的不能重载。

void foo(const int a);
void foo(int a);
//C语言中,参数修饰为const和不使用const修饰 被编译器一样对待,C++为兼容C,也使用了这个策略。因此二者等价。

如果一个成员函数在逻辑上不会修改对象的状态(字段),就应该定义为const函数

在上面的Dog类代码中,如果去掉bark函数后的const修饰符,并试着用一个const 对象去调用bark函数,则发现编译器报错。

void bark()
{
     for (size_t i = 0; i < energe; i++)
         cout << "wang wang!\n";
}

/////////////////////////
const Dog dog;

dog.bark(); //错误提示为类型不兼容

抛开代码,从业务逻辑上来看,bark函数只是在屏幕上输出消息,根本不会改变对象的状态,那即便是const 对象,也必须能成功调用啊。

这里报错显然是bark函数的问题:应该定义为const函数,这点大家都是很清楚的。但是为什么要这样的呢?

按照最开始介绍的分析方法,发生错误的代码等价与下面的C代码。

const Dog dog;

bark(&dog)      //转化后的bark的原型:void bark(Dog*this);

显然,将常量的指针赋值给非常量指针是不允许的。

再比如,C++标准库中的string类,它的用于获取字符串长度的成员函数都是const修饰的。如果不是这样的话,那么字符串常量就不能获取他们的长度了,这简直荒谬!

size_type size() const;

size_type length() const;

const成员函数可以形成重载

成员函数同时存在 const版本和非const版本?可以重载?是的。

同样,先按照最开始介绍的第1条分析方法,转换为C代码。然后根据第3条的分析就可以了。这里不再赘述。

最常用的就是,按照约定当重载索引运算符 [ ] 时,会同时编写一个 const版本和非const版本。

例如std::vector的[ ]运算符函数

reference       operator[]( size_type pos );

const_reference operator[]( size_type pos ) const;

总结

1、如果一个成员函数在逻辑上不会修改对象的状态(字段),就应该定义为const函数

2、

如果对象是const,则它只能调用const成员函数。

如果对象是普通的非const对象:

调用的某个成员函数是非const函数,则理所当然调用它。

调用的某个成员函数是const函数,则当然也可以调用他。(底层const指针可以指向非常量对象)

调用的某个成员函数同时存在 const版本和非const版本,则优先调用非const成员函数,编译器总是使用最匹配的版本。

时间: 2024-10-14 06:43:52

C++ const成员函数的相关文章

拷贝构造函数和const成员函数

实验原因 说明如何使用const描述保护类数据不会意外修改. 编译环境 vc6sp6 + win7x64 工程下载 copyConstruction_constMemberFunction.zip 使用非const成员函数,引起的拷贝构造函数报错 [cpp] view plain copy class CStudent { /// 常量定义 public: enum {NAME_SIZE_MAX = 64}; /// 构造, 拷贝构造, 析构函数 public: CStudent(); CStu

const成员函数

本文结论: const对象.指向const对象的指针或引用只能用于调用其const成员函数. 实例说明: class A { public: void mf1(){ cout<<"Function Call"<<endl; } void mf2() const{ cout<<"const Function Call"<<endl;} }; int main() { A a; a.mf1(); //ok a.mf2();

C++类const成员函数

c++类中可以用const关键字来声明不修改对象的成员函数,它使用时需要注意: 1.常量对象只能调用const成员函数 2.非常量对象可以调用const成员函数 以下列代码为例,来解释: 1 class Person{ 2 public: 3 void print(){ 4 cout<<"I am a common function!"<<endl; 5 } 6 void show()const{ 7 cout<<"I am a const

mutable用于修改const成员函数中的成员变量

http://no001.blog.51cto.com/1142339/389840/ mutalbe的中文意思是“可变的,易变的”,跟constant(既C++中的const)是反义词. 在C++中,mutable也是为了突破const的限制而设置的.被mutable修饰的变量,将永远处于可变的状态,即使在一个const函数中. 我们知道,如果类的成员函数不会改变对象的状态,那么这个成员函数一般会声明成const的.但是,有些时候,我们需要在const的函数里面修改一些跟类状态无关的数据成员,

C++ Primer 学习笔记_24_类与数据抽象(10)--static 与单例模式、auto_ptr与单例模式、const成员函数、const 对象、mutable修饰符

C++ Primer 学习笔记_24_类与数据抽象(10)--static 与单例模式.auto_ptr与单例模式.const成员函数.const 对象.mutable修饰符 前言 [例]写出面向对象的五个基本原则? 解答:单一职责原则,开放封闭原则,依赖倒置原则,接口隔离原则和里氏替换原则 里氏替换原则:子类型必须能够替换他们的基类型. 设计模式分为三种类型:创建型模式.结构型模式和行为型模式 一.static 与单例模式 1.单例模式 单例模式的意图:保证一个类仅有一个实例,并提供一个访问它

(转)函数后面加const--C++ const成员函数

类的成员函数后面加 const,表明这个函数不会对这个类对象的数据成员(准确地说是非静态数据成员)作任何改变. 在设计类的时候,一个原则就是对于不改变数据成员的成员函数都要在后面加 const,而对于改变数据成员的成员函数不能加 const.所以 const 关键字对成员函数的行为作了更加明确的限定:有 const 修饰的成员函数(指 const 放在函数参数表的后面,而不是在函数前面或者参数表内),只能读取数据成员,不能改变数据成员:没有 const 修饰的成员函数,对数据成员则是可读可写的.

c++中的const参数,const变量,const指针,const对象,以及const成员函数

const 是constant 的缩写,“恒定不变”的意思.被const 修饰的东西都受到强制保护,可以预防意外的变动,能提高程序的健壮性.所以很多C++程序设计书籍建议:“Use const whenever you need”. 1.用const 修饰函数的参数 如果参数作输出用,不论它是什么数据类型,也不论它采用“指针传递”还是“引用传递”,都不能加const 修饰,否则该参数将失去输出功能.const 只能修饰输入参数: 如果输入参数采用“指针传递”,那么加const 修饰可以防止意外地

C++中const对象和const成员函数

1?  成员函数可以声明成const函数(声明后加const) 2?  对于const对象,只能调用const成员函数 3?  Const函数和非const函数可以形成重载 4?  对于非const对象的函数调用优先选择非const成员函数 5?  对于类中的mutable数据成员,可以被const成员函数修改 // // main.cpp // Const // // Created by 06 on 15/1/24. // Copyright (c) 2015年 黄永锐. All right

关于 const 成员函数

成员函数如果是const意味着什么? 有两个流行概念:物理常量性和逻辑常量性. C++对常量性的定义采用的是物理常量性概念,即const 成员函数不可以更改对象内任何non-static成员变量.例如: 1 class CTextBlock 2 { 3 public: 4 ...... 5 std::size_t length() const; 6 7 private: 8 char* pText; 9 std::size_t textLength; 10 bool lengthIsValid;

const 成员函数

我们知道,在成员函数中,如果没有修改成员变量,应该给成员函数加上 const 修饰符,例如 1 #include <iostream> 2 3 using namespace std; 4 5 class Foo 6 { 7 public: 8 Foo(int x) : _x(x) {} 9 int getX() const { return _x; } 10 private: 11 int _x; 12 }; 13 14 int main() 15 { 16 Foo f(1); 17 cou