在C/C++中,经常会发生数据类型转换,例如整型数据可以赋值给浮点型变量,在赋值之前,先把整型数据转换为浮点型;反过来,浮点型数据也可以赋值给整型变量。

数据类型转换的前提是,编译器知道如何对数据进行取舍。例如:

int a = 10.9;
printf("%d\n", a);

输出结果为 10,编译器会将小数部分直接丢掉(不是四舍五入)。再如:

float b = ;
printf("%f\n", b);

输出结果为 10.000000,编译器会自动添加小数部分。

类也是一种数据类型,也可以发生数据类型转换。不过这种转换只有在基类和派生类之间才有意义。

由于派生类包含从基类继承的成员,因此可以将派生类的对象赋值给基类对象,如下所示:

class A{
public:
int m;
};
class B: public A{
public:
int n;
};
A a;
B b;
a = b;
a.m = ;

A是B的基类,a、b 分别是它们的对象,所以可以将 b 赋值给 a,然后通过 a 来访问成员变量 m。

实际上,对象之间的赋值是成员变量的赋值,成员函数不存在赋值问题。在赋值时,会舍弃派生类自己的成员,也就是”大材小用“,如下图所示:

可以发现,即使将派生类对象赋值给基类对象,基类对象也不会包含派生类的成员,所以依然不同通过基类对象来访问派生类的成员。对于上面的例子,a.m 是正确的,但 a.n 就是错误的,因为 a 不包含成员 n。

这种转换关系是不可逆的,只能用派生类对象给基类对象赋值,而不能用基类对象给派生类对象赋值。理由是很显然的,基类不包含派生类的成员,无法对派生类的成员变量赋值。同理,同一基类的不同派生类对象之间也不能赋值。

要理解这个问题,还得从赋值的本质入手。赋值实际上是向内存填充数据,当数据较多时很好处理,舍弃即可;上例中,a=b 时,成员 n 是多余的,会被直接丢掉,所以不会发生赋值错误。但当数据较少时,问题就很棘手,编译器不知道如何填充剩下的内存;上例中,b= a 时,编译器不知道该如何给变量 n 赋值,所以会发生错误。

请看下面一个完整的例子:

#include <iostream>
using namespace std;
class A{
public:
int n;
A(int n):n(n){}
void display(){ cout<<"Class A: n="<<n<<endl;}
};
class B: public A{
public:
B(int n):A(n){}
void display(){ cout<<"Class B: n="<<n<<endl;}
};
class C: public A{
public:
C(int n):A(n){}
void display(){ cout<<"Class C: n="<<n<<endl;}
};
int main(){
A a();
B b();
C c();
a.display();
a = b;
b.n = ;
a.display();
a = c;
a.display();
return ;
}

本例中,将 b 对象赋值给 a 对象,等价于 a.n = b.n,赋值完成后 a.n 的值为 2,然后调用 a.display() 函数,输出结果就是”Class A: n=2“。将 c 对象赋值给 a 对象也是同样的道理。

这个例子很好的说明了:基类对象和派生类对象之间的赋值仅仅是对应的成员变量的赋值,不会影响成员函数,不会影响 this 指针。

总结:可以通过派生类对象访问派生类的成员,但无论如何,也不能通过基类对象访问派生类成员。

指向对象的指针

对上例 main 函数中的代码做如下更改:

A *p = new A();
p->display();
p = new B();
p->n = ;
p->display();
p = new C();
p->display();

输出结果:
Class A: n=1
Class A: n=100
Class A: n=3

本例定义了一个指针,使它指向不同的对象。与上例不同的是,本例中并没有发生对象的赋值,仅仅是改变了指针的指向。

输出结果的第2行与上例不同,这是为什么呢?

在代码第3行中,将 p 指向 B 类的对象,隐式指针 this 也随之改变,指向 B 类的对象,所以 p->n 和 this->n 访问的都是 B 类对象的成员变量,输出结果显然是”n=100“。

细心读者可能已经发现,虽然 this 指向了 B 类对象,但是 p->display() 依然调用 A 类的成员函数。这是因为,成员变量和成员函数不在同一个内存区域,系统通过 this 指针来访问成员变量,但是却不通过它来访问成员函数。

系统通过对象的类型来访问成员函数。例如 p 的类型是 A,那么不管 p 指向哪个对象,都访问 A 类的成员函数。

C++学习21 基类和派生类的赋值的更多相关文章

  1. 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)

    [源码下载] 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成 ...

  2. C++学习笔记(6)----基类和派生类的构造函数和析构函数的执行顺序

    基类和派生类:构造函数和析构函数的执行顺序 在Visual Studio中,新建控制台工程,构造类如下: #include<iostream> using namespace std; c ...

  3. (转) C++中基类和派生类之间的同名函数的重载问题

    下面有关派生类与基类中存在同名函数 fn: class A { public: void fn() {} void fn(int a) {} }; class B : public A { publi ...

  4. 详解C++中基类与派生类的转换以及虚基类

    很详细!转载链接 C++基类与派生类的转换在公用继承.私有继承和保护继承中,只有公用继承能较好地保留基类的特征,它保留了除构造函数和析构函数以外的基类所有成员,基类的公用或保护成员的访问权限在派生类中 ...

  5. c++中基类与派生类中隐含的this指针的分析

    先不要看结果,看一下你是否真正了解了this指针? #include<iostream> using namespace std; class Parent{ public: int x; ...

  6. 基类和派生类--this

    基类指针在程序运行的时候的确指向的是一个派生类的对象,但指针的类型仍然是基类指针.C++是一种强类型语言,因此不能用基类指针类型的指针直接调用派生类:而且,同一个类可能有多种不同的派生类,因此不知道实 ...

  7. C++:基类与派生类对象之间的赋值兼容关系

    4.5 基类与派生类对象之间的赋值兼容关系 在一定条件下,不同类型的数据之间可以进行类型转换,例如可以将整型数据赋给双精度型变量. 在赋值之前,先把整型数据转换为双精度型数据,然后再把它双精度型变量. ...

  8. C++:基类和派生类

    4.1 派生类的声明 继承实例如下: class Person{ //声明基类Person public: void print() { cout<<"name:"&l ...

  9. C++基类和派生类之间的转换

    本文讲解内容的前提是派生类继承基类的方式是公有继承,关键字public 以下程序为讲解用例. #include<iostream> using namespace std; class A ...

随机推荐

  1. ArrayList、linklist、list的区别

    List是一个接口,ArrayList和LinkedList是两个实现类,他们实现的方式不一样,其实LinkedList才是真正的链表(如果不清楚什么是链表,需要了解一下相关数据结构的知识,这不是一两 ...

  2. 虚拟化之vmx配置文件

    https://www.haiku-os.org/get-haikuhttp://sanbarrow.com/vmx.html contradictionn. 矛盾:否认:反驳homegrownadj ...

  3. unity, Animation crossfade需要两动画在时间上确实有交叠

    unity现在播动画都用Animator了,但公司的老项用的还是Animation,今天遇到一个bug,是两个动画的衔接处不连贯. 最后发现是由于A动画已经播完之后B动画才开始播,而且还用了cross ...

  4. BigPipe 了解

    BigPipe是一个重新设计的基础动态网页服务体系.大体思路是,分解网页成叫做Pagelets的小块,然后通过Web服务器和浏览器建立管道并管理他们在不同阶段的运行.这是类似于大多数现代微处理器的流水 ...

  5. 彻底解决ASP.NET MVC 3 404错误码返回302的问题

    转自:http://blog.csdn.net/mycloudke/article/details/9746333 404状态码:,意味着当在页面上显示用户点击不存在,提高用户体验度,搜索引擎会放弃这 ...

  6. 概要设计、详细设计(三)关键点(Know-How)、运用技巧

    1.    关键点(Know-How).运用技巧 4.1 设计准则 制定设计准则是概要设计阶段的最主要.最关键的工作.在实际工作中往往忽略,多数项目牺牲在这个环节.制定设计准则着眼于如何更好的做设计, ...

  7. LinkedHashMap和HashMap区别

    import java.util.HashMap; import java.util.Iterator; import java.util.LinkedHashMap; import java.uti ...

  8. Android 不同文件名介绍

    Android 不同文件名介绍

  9. Jar mismatch! Fix your dependencies的问题

    在开发Android项目的时候,有时需要引用多个项目作为library.在引用项目的时候,有时会出现“Jar mismatch! Fix your dependencies”错误. 这是因为两个项目的 ...

  10. PHP中使用CURL实现get和post请求(总结)

    一.什么是curl curl是利用url语法在命令行方式下工作的开源文件传输工具. 二.PHP curl函数