怎么成了抄书了,雷神也不知不觉,可能是在这章的理解上比较容易些吧,不用去想个看的见摸的着的东西比划。好象小朋友学算术,一位数的计算不用掰手指头,可是两位数或者三位数的计算,手指头加上脚指头还是不够。学习就是这么回事。理解力和抽象能力很重要。回来继续学习。
通过这一章我还知道了。数据成员的布局。数据成员的存取。并且对Static data members有了进一步的了解,在class的生命周期中,静态成员被看作是全局变量,每一个member的存取不会导致任何空间或效率上的额外负担。不论是从一个复杂的继承关系中继承还是直接声明的,Static data member都只会有一个实体。并且有着非常直接的存取路径。另外如果两个类都声明了一个相同名字的静态成员变量,那么编译器会通过一种算法,为我们解决名字冲突的问题。而非静态的成员变量的存去实际上是通过implicit class object(this指针)来完成的。例如
Point3d Point3d::translate(const Point3d &pt) { x+=pt.x; y+=pt.y; z+=pt.z; } 被编译器经过内部转换成为了下面这个样子: Point3d Point3d::translate(Point3d *const this,const Point3d &pt) { this->x+=pt.x; this->y+=pt.y; this->z+=pt.z; }
如果要对一个非静态的成员变量进行存取,编译器会把类对象的起始地址加上数据成员的偏移量。例如:
Point3d origin; origin._y=0.0; //地址&origin._y将等于 &origin+(&Point3d::_y-1); 目的是使编译系统能够区分出以下两种情况: 一个指向数据成员的指针,用来指出类的第一个成员。 一个指向数据成员的指针,没有指出任何成员。 这是什么意思?什么是指向数据成员的指针。书上的例子: class Point3d { public: virtual ~Point3d(); //…… protected: static Point3d origin;//静态的数据成员,位置在class object之外 float x,y,z;//每个float是4bytes } &Point3d::z; //这个值是什么?
我们在这篇文章开始的时候已经知道了还有一个vptr,不过vptr的位置也许在对象的开始,也许在对象的结尾部。所以上面的操作的值应该是8或者12(如果vptr在前面的话)。但实际上取会的值被加上了1。原因是必须要区别一个不指向任何成员的指针,和一个指向第一个成员的指针。又有点不好理解了,举个例子:
想象你和你的另外两个朋友合住一个三室一厅的房子,你住在第一间。如果你给一个你们三个人共同的朋友的地址你可以给房号就行了。不用给出你们的任意一个人的那间房子号(不指向任何成员)。但如果你给你的一个私人朋友地址,你会给出房间号和你的那个房间号。为了使这个地址有区别,你必须有一个厅来作为偏移量(offset)。不知道大家明白这个例子吗,也许这个例子会影响你的正确思维。那就太糟糕了。不过我还是喜欢这样想问题,也许不太准确,但可以帮助我,因为想象一个内存空间比想象一个三居室要难好几点儿。  
2/2 首页 上一页 1 2 |