【C++】 私有成员变量的理解

私有成员变量的概念,在脑海中的现象是,以private关键字声明,是类的实现部分,不对外公开,不能在对象外部访问对象的私有成员变量.

然而,在实现拷贝构造函数和赋值符函数时,在函数里利用对象直接访问了私有成员变量,因而,产生了困惑.下面以具体实例进行说明:

疑惑:为什么第26行和第32行代码可以编译通过,而第39行和第40行代码会产生编译错误?

 1 class CTest {
 2 public:
 3     CTest(int i);
 4     CTest(const CTest& rhs);
 5     CTest& operator=(const CTest& rhs);
 6     void printCTest(const CTest& rhs);
 7 private:
 8     int value;
 9 };
10
11 CTest::CTest(int i):value(i)
12 {
13     cout<<"Contructor of CTest"<<endl;
14 }
15
16 CTest::CTest(const CTest& rhs):value(rhs.value)
17 {
18     cout<<"Copy contructor of CTest"<<endl;
19 }
20
21 CTest& CTest::operator=(const CTest& rhs)
22 {
23     cout<<"Assign function of CTest"<<endl;
24     if(this == &rhs)
25         return *this;
26     value = rhs.value;                //通过对象访问私有成员变量
27     return *this;
28 }
29
30 void CTest::printCTest(const CTest& rhs)
31 {
32     cout<<rhs.value<<endl;        //通过对象访问私有成员变量
33 }
34
35 int main()
36 {
37     CTest t = 1;
38     CTest tt = 2;
39     //  cout<<t.value<<endl;        //通过对象访问私有成员变量,编译错误
40     //  cout<<tt.value<<endl;        //通过对象访问私有成员变量,编译错误
41     t.printCTest(tt);
42 }  

产生这种疑惑的原因是自己对私有成员变量的理解有误,封装是编译期的概念,是针对类型而非对象的,在类的成员函数中可以访问同类型实例对象的私有成员变量

具体的解析如下:从变量value的符号是怎么解析的分析

1.确定符号的查找域

如第26行代码,当编译器发现value变量时,它会在value变量所属的对象rhs的类域中寻找该符号.

2.确定当前域中哪些符号可以访问

由第1步可知,当前查找的域是类域,而printCTest函数在CTest类体中,所以printCTest可以访问CTest类中的所有变量(包括私有成员变量),因而value符号在CTest类域中被找到.

如第39行代码,main函数不在CTest类体中,所以main函数不可以访问CTest类域中的私有成员变量.

3.符号已查找到,编译通过

类成员变量的访问权限是编译器强加的,编译器可以找到value,通过编译,自然就可以访问到value变量的值.

直觉上,我们会以为第26行代码中value符号的查找域应该是对象rhs对应的作用域,然而C++编译器的实现却是在对象rhs的类域查找value符号.

启发:有些直觉是靠不住的,需要深入分析其背后的实现原理,才可以理解透彻.

时间: 2024-10-28 02:59:45

【C++】 私有成员变量的理解的相关文章

[C++参考]私有成员变量的理解

私有成员变量的概念,在脑海中的现象是,以private关键字声明,是类的实现部分,不对外公开,不能在对象外部访问对象的私有成员变量. 然而,在实现拷贝构造函数和赋值符函数时,在函数里利用对象直接访问了私有成员变量,因而,产生了困惑.下面以具体实例进行说明: 疑惑:为什么第26行和第32行代码可以编译通过,而第39行和第40行代码会产生编译错误? class CTest { public: CTest(int i); CTest(const CTest& rhs); CTest& opera

通过反射,如何操作私有成员变量(取/赋值),如何调用私有方法?

Java的反射工具很强大,有句著名的话:No reflection ,no frameworks. 工作中直到涉及到UT,才体会到它的重要性,现归纳整理一个小例子: 反射工具类: 1 import java.lang.reflect.Field; 2 import java.lang.reflect.InvocationTargetException; 3 import java.lang.reflect.Method; 4 5 public class ReflectionUtil { 6 7

JavaSE7基础 通过公有的成员方法 去过滤 要赋给私有成员变量的值

版本参数:jdk-7u72-windows-i586注意事项:博文内容仅供参考,不可用于其他用途. 代码 class Test{ private int age;//只能在本类中访问 //公有的成员方法 去过滤 要赋给私有成员变量的值 public void setAge(int age){ if(age<0 || age>120){ System.out.println("年龄赋值有误"); this.age=0; }else { System.out.println(&

Hibernate通过什么方法可以把私有成员变量赋值成数据库查询到的值然后返回POJO对象呢?

public void setAccessible(boolean flag) throws SecurityException将此对象的 accessible 标志设置为指示的布尔值.值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查.值为 false 则指示反射的对象应该实施 Java 语言访问检查.参考博文:http://www.cnblogs.com/sunxucool/p/3552985.html Hibernate通过什么方法可以把私有成员变量赋值成数据库查询到

C++私有成员变量被对象直接访问

C++访问控制: 关键字private,它描述了对类成员的访问控制.使用类的对象方法可以直接访问私有成员函数和变量.类的对象即实例只能通过公共成员函数来访问私有变量和私有成员函数.因此公有成员函数成为对象的私有变量访问的桥梁. 在c++中,在类的成员函数的参数为此类类型时,可以通过类类型的对象直接访问私有成员变量.或者在成员函数中使用临时对象来直接访问私有变量. class test { private: int a ; int b ; public: test(int x,int y) { a

私有成员变量可以被子类继承吗

今天看spring源码,不经意间发现了一问题:在一个抽象类中定义了一个私有成员变量. 仔细想想:抽象类不能被实例化的,只能被子类继承.但是自从学java的继承只有,我们就知道,子类不能继承父类的私有成员变量或方法的. 问题:在该抽象方法中定义这个私有变量有什么用呢?或者说这个私有成员变量再什么地方用得到呢? 所以自己做了一个测试如下: public abstract class Fatherclass { private int privatenumber; public int getPriv

关于java私有成员变量的访问

java 主类M中可以直接访问自身的私有成员,就算在main中 new 一个自己,也只是自身静态方法中的一个局部变量,该局部变量依然可以访问自身的private成员,只要该局部变量身处自身所在类M. 主类M外面定义了其它类A,然后在主类的main方法中编写如下 语句 : 1.M m = new M ,那么可通过 m.p 直接访问主类私有成员,因为该语句身处M类体中 2.A a = new A ,那么不能通过 a.p直接访问私有成员,因为该语句身处M类体中,不在A类体中 读大学时学C++的惯性思维

c++中类对象不能访问类的私有成员变量

类的成员变量分三种: public   protect  private 1.类的成员函数可以访问类这三类 2.继承类成员函数可以访问 protect  类型 3.只有本身类的成员函数才可以访问 private 类型 4.类的对象若在其他函数中是不可以访问类的 protect  private类型变量 类其实是个模型.而对象就是找个这个模型制作的实体.比如,类是造车的图纸,对象就是找个图纸造出的汽车.类的成员是分级别的:protect,public,private.在类内部定义的函数可以访问任何

Java子类访问父类的私有成员变量

/**子类会继承父类所有的属性和方法. * 但是根据不同的权限标识符,子类不可见父类的私有变量,但可以通过父类的公共方法访问私有变量 * 所以对于重名变量,子类和父类都各有一份. * 对于子类和父类中重名的方法,则为重写.即子类重写了父类的方法,用于多态. * 同一个类中函数的签名不同,则为方法的重载.函数的签名为函数名+参数列表,与返回值无关. */