深入理解.net - 1.继承的本质
最近偶然看到这个博客你必须知道的.net,作者6的飞起啊,干货十足,还是07年写的。。。写的也很赞,评论更精彩,在此强烈推荐一波,看的感觉就像沙漠里发现了绿洲一样,很兴奋,意犹未尽,迫不及待的看完一篇再看下一篇,但是知识还是需要整理,沉淀的,那就写博客吧,于是有了接下来的文章。本文将通过看此书和相关博客以及结合自己目前的理解所写,如有不对之处,欢迎指正。
对象的创建过程
要了解继承的本质首先我们要清楚一个对象的创建过程,这里有个 Chicken 类:
public class Chicken
{
private string type = "Chicken";
public Chicken()
{
}
public void ShowType()
{
Console.WriteLine($"Type is {type}");
}
}
当我们需要使用这个类的时候,我们通常是这样写的:
Chicken chicken = new Chicken();
它是如何工作的呢?先上图:

具体过程如下:
- 首先执行的是 "Chicken chicken" 语句,即线程栈Stack上声明了一个Chicken类型的引用chicken,此时值为null,Stack上内存分配由高到低地址开始创建, 而Heap上则相反;
- 执行 "new Chicken()" ,new 操作符会在托管堆(具体在GCH:Garbage Collection Heap)上申请创建实例的内存空间,初始化类的字段(Feild)信息,并调用构造函数。结合上图,实例在GCHeap创建的详细过程如下:
- 对象实例地址的开始4个字节为SyncBlockIndex,指向SyncEntryTable,存储的是多线程同步的一些信息,详细内容可查看文章末尾参考连接;
- 紧接着是TypeHandle,指向的是Loader Heap(加载器堆) 中的MethodTable,而MethodTable中存储该类型的静态字段,方法表以及实现的接口等信息,从这里我们也就清楚了,一个类不管实例成员有多少,static成员和方法信息只存储一份在内存中,并先于实例创建,使用的时候则通过TypeHandle到MethodTable查找,并编译成cpu指令,存储在内存中,以后再使用时则直接执行该指令即可。
- 初始完SyncBlockIndex和TypeHandle,则加载Chicken类型的字段信息,本文初始的也就是type字段(字符串信息的存储比较特殊实际存储模型详见此链接),另外强调的是属性不在此处初始,属性本质上还是 _Get/_Set方法;
- 初始完字段后,则调用构造函数Chicken(),并返回this。
3.最后将this赋值给Stack上的chicken引用类型,即chicken维护一个指向heap上Chicken实例的指针,实际stact上的chicken存储的是GCHeap上实例存储的地址;
继承的本质
如果你看到这里,那说明你已经对一个对象的创建过程有了清晰的认识。回归主题那继承的本质是什么?先别急,下面我们写一个 Animal 类,让上文中的Chicken类继承它,并重写父类中的ShowType方法,本示例代码参考书中示例略微有所调整代码如下:
public class Animal
{
private string type ="Animal";
public Animal()
{
}
public virtual void ShowType()
{
Console.WriteLine($"Type is {type}");
}
}
public class Chicken : Animal
{
public string type = "Chicken";
public Chicken()
{
}
public override void ShowType()
{
Console.WriteLine($"Type is {type}");
}
}
那么这个时候我们去执行 Chicken chicken = new Chicken(); 发生了什么呢?

根据上图我们可以很直观的看出(此处暂时不考虑Object):
- 首先会先初始化chicken.type 字段,然后调用Chicken 构造函数;
- 此时编译器发现还有父类则去为父类Animal 申请内存,即初始Animal.type 字段,然后调用Animal的构造函数;因为所有类型都是继承自System.Object 所以实际上会一直遍历到Object类型;此外从这个过程中我们也可以发现子类是可以继承父类私有成员信息,即chicken可以继承Animal的type字段,字段存储顺序是父类在前子类在后,跟踪截图如下:

- Animal()方法体执行完后,然后在执行Chicken()的方法体。
- 此处额外说下关于方法的加载,在继承过程子类会将父类中的方法copy一份,并将重写的方法覆盖掉父类中的方法,这也就为多态提供了基础。
最后
写这篇博客参考了不少其它牛人的博客,发现关于这块往深里东西还有很多,如AppDomain应用程序域,ManagerHeap可以分多种不同的类型,GC对不同的Heap处理规则也是不同的,近期也会持续分享相关内容。写博文期间内容也不断反复调整了几轮,希望在此我都表达清楚了,限于篇幅主要内容还是关于对象和继承的本质过程,内容基本上也都是根据自己的理解写出来的,难免有疏漏的地方,如有不对的对方还请指出,那将是我不断进步的源泉:-)。
参考
- 《你必须知道的.net(第2版)》 - 王涛
- 你必须知道的.net博客目录:http://www.cnblogs.com/anytao/archive/2007/09/14/must_net_catalog.html - 王涛
- 类型实例的创建位置、托管对象在托管堆上的结构 - Silent Void
- 关于CLR内存管理一些深层次的讨论下篇 - Artech
深入理解.net - 1.继承的本质的更多相关文章
- 个人理解java的继承
java的类是属于单继承的.在继承这一块上我本来有一个很大的误区,就是觉得父类中private定义的成员无法被继承.直到网上的大神给我指出private是可以被继承的,会在内存中,只是在子类的对象中不 ...
- 彻底理解Javascript原型继承
彻底理解Javascript原型继承 之前写过一篇Javascript继承主题的文章,这篇文章作为一篇读书笔记,分析的不够深入. 本文试图进一步思考,争取彻底理解Javascript继承原理 实例成员 ...
- 【CUDA 基础】3.2 理解线程束执行的本质(Part I)
title: [CUDA 基础]3.2 理解线程束执行的本质(Part I) categories: CUDA Freshman tags: 线程束分化 CUDA分支 toc: true date: ...
- c++ 基础知识回顾 继承 继承的本质就是数据的copy
c++ 基础知识笔记 继承 什么是继承 继承就是子类继承父类的成员属性以及方法 继承的本质就是 数据的复制 是编译器帮我们做了很多操作 class Base { public: Base(){ cou ...
- 从一些简单代码实例彻底理解面向对象编程思想|OOP本质是什么?
从Rob Pike 的 Google+上的一个推看到了一篇叫<Understanding Object Oriented Programming>的文章,我先把这篇文章简述一下,然后再说说 ...
- java四大特性理解(封装继承多态抽象)
封装: 封装是把过程和数据包围起来,对数据的访问只能通过已定义的接口.面向对象计算始于这个基本概念,即现实世界可以被描绘成一系列完全自治.封装的对象,这些对象通过一个受保护的接口访问其他对象.封装是一 ...
- 27、理解js的继承机制(转载自阮一峰)
Javascript继承机制的设计思想 作者: 阮一峰 日期: 2011年6月 5日 我一直很难理解Javascript语言的继承机制. 它没有"子类"和"父类&qu ...
- c++父类指针强制转为子类指针后的测试(帮助理解指针访问成员的本质)(反多态)
看下面例子: #include "stdafx.h" #include <iostream> class A { //父类 public: void f() / ...
- 理解oo:继承、多态、重写、重载、接口、抽象类
1. 继承: 从多个子类中抽象出实例变量以及方法,形成更抽象的父类,避免在子类中的代码重复,维护起来更加方便.检查是否可以使用继承技术的方法是:IS A 对于类A继承自类B,类C继承自类A,那么类C和 ...
随机推荐
- MyBatis之ResultMap简介,关联对象
MyBatis中在查询进行select映射的时候,返回类型可以用resultType,也可以用resultMap,resultType是直接表示返回类型的,而resultMap则是对外部ResultM ...
- 通过jersey-client客户端调用Jersey的Web服务模拟CURD
一.总体说明 通过jersey-client接口,创建客户端程序,来调用Jersey实现的RESTful服务,实现增.删.改.查等操作. 服务端主要是通过内存的方式,来模拟用户的增加.删除.修改.查询 ...
- 谷歌、腾讯、百度相应API批量获取地理位置坐标信息及其优缺点
目录: 申请ak 批量获取地理位置 目的:通过给定的地理位置名称(如:北京市海淀区上地十街十号),获取经纬度信息. 1.申请ak 以百度Geocoding API为例:http://lbsyun.ba ...
- 以太坊开发DApp实战教程——用区块链、星际文件系统(IPFS)、Node.js和MongoDB来构建电商平台(一)
第一节 简介 欢迎和我们一起来用以太坊开发构建一个去中心化电商DApp!我们将用区块链.星际文件系统(IPFS).Node.js和MongoDB来构建电商平台类似淘宝的在线电商应用,卖家可以自由地出售 ...
- [COGS 2583]南极科考旅行
2583. 南极科考旅行 ★★ 输入文件:BitonicTour.in 输出文件:BitonicTour.out 简单对比时间限制:1 s 内存限制:256 MB [题目描述] 小美要 ...
- spring框架学习笔记4:SpringAOP实现原理
AOP AOP(Aspect Oriented Programming),即面向切面编程,可以说是OOP(Object Oriented Programming,面向对象编程)的补充和完善.OOP引入 ...
- 2017年秋软工-领跑衫获奖感言&我最感谢的人
啥都不说,先上幅图.获得领跑衫,开心. 一.回忆 这是我第二次来上恩师杨的软件工程,第一次是2016年春,那时候我还是本科三年级的学生.忘了第一次为啥去蹭课,印象中是我的榜样亮哥把我给忽悠过去的?我也 ...
- Week02-Java基本语法与类库
1. 本周学习总结 本周主要学习了包装类,明白了包装类比基本数据类型的优点(见书面作业4.2),了解了自动装箱,自动拆箱的概念和区别(见书面作业4.1),知道了java中的引用类似C语言中的指针,明白 ...
- Scapy实现SYN泛洪攻击
一.实验说明 1.实验介绍 本次实验将使用python3版本的Scapy--Scapy3k来实现一个简单的DDos,本次实验分为两节,本节将学习如何使用Scapy3k来实现SYN泛洪攻击. 2.知识点 ...
- Python科学计算(一)
作者 J.R. Johansson (robert@riken.jp) http://dml.riken.jp/~rob/ 最新版本的 IPython notebook 课程文件 http://git ...