-----------------siwuxie095
常对象成员 和 常成员函数
看如下实例:
定义一个坐标类:Coordinate
要想正确的初始化,必须通过初始化列表来初始化,因为两个数据成员
都是由 const 来修饰的,是常数据成员
作为一个类的数据成员来说,是可以用 const 来修饰的
显然,对象作为数据成员,也是可以用 const 来修饰的,称之为 常对象成员
以线段为例:当线段的位置一旦确定,就不能再更改
定义一个线段类:
因为要实现两个点一旦初始化,就不能再修改,需要定义成 const
在构造函数中通过初始化列表来初始化:
在使用时:
const 不仅可以修饰数据成员,还可以修饰成员函数
const 修饰的成员函数,称之为 常成员函数
「常成员函数中,const 修饰的实际上就是 this 指针」
看如下实例:
定义 changeX() 函数:上面的写法是错误的,下面的写法是正确的
常成员函数中为什么不能修改数据成员的值呢?
当定义 changeX() 函数时,看上去没有任何的参数,而实际上
却隐含着一个参数:this 指针
当成员函数不是一个普通的成员函数,而是一个 const 修饰过的
常成员函数时,依然有一个隐含的参数:this 指针,但 this 指针
是用 const 来修饰的,是一个常指针。通过常指针去改变该指针
指向的数据,肯定是不被允许的
所以,如果在常成员函数中去修改数据成员的值,这种做法就一定是错误的
此外,changeX() 和 changeX() const 二者互为重载
虽然从语法的角度来说,这个的确可以有,但如果真要去这么定义的话,
那我还是挺佩服你的,就像佩服孔乙己知道 回 字有 4 种写法一样
因为这样定义,接下来在使用时,肯定会感到疑惑:调用的是哪个成员
函数?实际上调用的是不带 const 的普通的成员函数 changeX()
那么在什么情况下可以调用常成员函数?
在实例化对象时,必须用 const 来修饰这个对象,这样实例化的
对象,称之为 常对象
通过常对象调用的 changeX(),就是用 const 修饰的常成员函数
程序:
Coordinate.h:
class Coordinate { public: Coordinate(int x,int y); ~Coordinate(); void setX(int x); //const修饰后只具有只读权限所以可以修饰get函数 //相当于 int getX(const Coordinate *this); int getX() const; //而set函数具有写权限所以不能用const(不能修改函数里面的值)修饰 void setY(int y); int getY() const; private: int m_iX; int m_iY; }; |
Coordinate.cpp:
#include "Coordinate.h" #include <iostream> using namespace std;
Coordinate::Coordinate(int x, int y) { m_iX = x; m_iY = y; cout << "Coordinate() " << m_iX << "," << m_iY << endl; }
Coordinate::~Coordinate() { cout << "~Coordinate() " << m_iX << "," << m_iY << endl; }
void Coordinate::setX(int x) { m_iX = x; }
int Coordinate::getX() const { return m_iX; }
void Coordinate::setY(int y) { m_iY = y; }
int Coordinate::getY() const { return m_iY; } |
Line.h:
#include "Coordinate.h"
class Line { public: Line(int x1,int y1,int x2,int y2); ~Line(); void setA(int x, int y); void setB(int x, int y); //常成员函数注意:const和括号之间要隔开一个空格不然有些编译器下可能无法通过 //相当于 void printInfo(const Coordinate *this); //也就是说常成员函数的const实际上修饰的是this指针使之成为常this指针 void printInfo() const; // void printInfo(Coordinate *this); 隐藏的this指针 //同名的普通成员函数和常成员函数互为重载 void printInfo();
private: const Coordinate m_coorA;//一个常对象成员 Coordinate m_coorB; //一个对象成员 }; |
Line.cpp:
#include "Line.h" #include <iostream> using namespace std;
Line::Line(int x1, int y1, int x2, int y2) :m_coorA(x1, y1), m_coorB(x2, y2) { cout << "Line()" << endl; }
Line::~Line() { cout << "~Line()" << endl; }
void Line::setA(int x, int y) { //m_coorA是常对象成员无法使用普通成员函数使用导致报错 //因为:setX(int x); 相当于 setX(Coordinate *this,int x); //传进去的是m_coorA的常this指针(只读)m_coorA是个只具有读权限的对象 //而原函数里的this既有读权限又有写权限 // //m_coorA.setX(x); //m_coorA.setY(y);
}
void Line::setB(int x, int y) { m_coorB.setX(x); m_coorB.setY(y); }
void Line::printInfo() const { cout << "printInfo() const" << endl; cout << "(" << m_coorA.getX() << "," << m_coorA.getY() << ")" << endl; cout << "(" << m_coorB.getX() << "," << m_coorB.getY() << ")" << endl; }
void Line::printInfo() { cout << "printInfo()" << endl; cout << "(" << m_coorA.getX() << "," << m_coorA.getY() << ")" << endl; cout << "(" << m_coorB.getX() << "," << m_coorB.getY() << ")" << endl; } |
main.cpp:
#include <stdlib.h> #include "Line.h" using namespace std;
int main(void) { Line line1(1, 2, 3, 4); line1.printInfo();//调用的是不带const的printInfo() const Line line2(5, 6, 7, 8); line2.printInfo();//调用的是带const的printInfo() system("pause"); return 0; }
//即常对象和常对象成员只能调用常成员函数 //而普通的对象和对象成员却可以调用普通成员函数和常成员函数 // //常数据成员和常对象成员必须使用初始化列表初始化 //(即const修饰的数据成员和对象成员在构造函数中只能用初始化列表初始化) //同名的普通成员函数和常成员函数互为重载 |
运行一览:
【made by siwuxie095】