今天看Lippman的《c++ primer》,解答了一直以来的疑惑。类的常量对象是无法访问非const成员函数的,即使你在函数中不对对象的数据成员进行更改。
请看下面一段代码~
#include <string>
using std::cout;using std::ostream;
using std::endl;using std::string;
class A{
public:
A(const string &str = "") :s(str){};
ostream &print(ostream &) const;
private:
string s;
};
ostream &A::print(ostream &o) const{
o << s;
return o;
}
int main()
{
A aObject("Hello!");
aObject.print(cout) << endl;
return 0;
}
我们知道,当对象在调用其成员函数的时候,其实是隐含传递了一个this指针的参数给成员函数的(不然该函数怎么知道是哪个对象在调用它^ ^)。默认情况下,这是一个const指针,指向一个非常量对象。这意味着我们不能将其绑定到一个常量对象。为什么呢?请看:
const string s1("I'm a const string!");
string s2("I'm not a const string1");
const string *p1;
string *p2;
p1 = &s1;//正确
p1 = &s2;//正确
p2 = &s1;//错误
p2 = &s2;//正确
想想为什么呢?
先不要看~~。。。。。。。
--------------------------------------------------------------------
我们定义p1是一个指向常量string的指针,意味着我们知道我们不会通过p1去更改p1指向的string的内容,那么p1指向的内容是否是常量就无所谓了,反正我们不会去修改它,对不?所以p1 = &s1;p1 = &s2;都是正确的!
但是对于p2,我们将其定义为一个指向非常量string的指针,那么我们可能会通过p2去修改p2指向的内容。这时候对于p2的赋值我们就要注意了,不能将一个const string的地址赋值给p2。试想一下,如果这样做的话~由于p2是一个指向非常量string的指针,我们可以修改它指向的string;可是它指向的string是一个const string啊,const string是不能被修改的,矛盾产生了。所以c++标准就不允许我们这样干啦!
---------------------------------------------------------------------------------------
好了,回到主题,我们已经知道我们无法将一个普通的this指针绑定到常量对象,所以常量对象就无法调用普通成员函数啊,普通成员函数需要一个普通的this指针(指向一个非常量对象),但你能给我的只有一个指向常量对象的this指针(因为你自己就是常量对象啊,只能给我这个了>>> <<<<),这两者是无法隐式转换的啊。那么怎么办呢?
只有霸王硬上弓了,通过在成员函数后面添加一个const修饰符,我们就将this声明为一个指向常量对象的常量指针啦!这下开心了,常量对象就可以将自己的this指针愉快的传给成员函数了,就 能调用它啦!
========================
总结就一句话,常量对象只能调用const成员函数