目录:

【C#小知识】C#中一些易混淆概念总结--------数据类型存储位置,方法调用,out和ref参数的使用

【C#小知识】C#中一些易混淆概念总结(二)--------构造函数,this关键字,部分类,枚举

【C#小知识】C#中一些易混淆概念总结(三)--------结构,GC回收,静态成员,静态类

【C#小知识】C#中一些易混淆概念总结(四)---------解析Console.WriteLine()

----------------------------------分割线--------------------------------------

这次主要分享的内容是关于继承的知识。

首先,我们先来看看继承;

既然有继承,就要有父类和子类,来看下面的一段代码:

class Person
{
private int nAge;
protected string strName;
double douHeight;
public string strEateType; public void Hello()
{
Console.WriteLine("我可以说Hello!");
}
public void Run()
{
Console.WriteLine("我可以跑!");
}
} class Student : Person
{ }

然后我在Main()函数中实例化子类的对象,代码如下:

  static void Main(string[] args)
{
Student stu1 = new Student();
}

那么在这个过程中内存中发生了些什么呢?

我们先来看misl的中间代码,看看那能发现些什么

由此我们可以发现子类继承了父类的所有成员包括Private和Protect,并为这些成员开辟了空间来存储。

我们再来实例化我们的子类,然后访问父类的字段和方法,会发现,如下的现象

所以虽然子类为父类的所有成员在堆中都开辟了空间,但是父类的私有成员(Private)子类访问不到,

而受保护的成员(protected)可以通过实例化对象访问的到。

所以在内存中的情况如下图:

看下面的代码,我们来探究一下在子类中this关键字和base关键字所访问的类的成员有哪些,代码如下:

class Student : Person
{
private string strClass; private string strAddress; public void Address(string cla, string adre)
{
//这里的this关键字调用了子类的成员和父类的非似有成员
this.strClass = "五";
this.strAddress = "北京";
this.strName = "子强"; //这里base关键字调用了是父类的非似有成员
base.strName = "强子"; Console.WriteLine("我是{0}年纪,来自{1}", cla, adre);
}
public void Sing()
{
this.strClass = "";
Console.WriteLine("我可以唱歌!");
}
}

所以在子类中this关键字和base关键字的访问范围的示意图如下:

二,关于子类对象的构造函数和父类构造函数的执行顺序

我们分别为父类和子类添加显式的构造函数,代码如下

class Person
{
private int nAge;
protected string strName;
double douHeight;
public string strEateType; //父类的构造函数
public Person()
{
Console.WriteLine("我是父类的构造函数");
} public void Hello()
{
Console.WriteLine("我可以说Hello!");
}
public void Run()
{
Console.WriteLine("我可以跑!");
} } class Student : Person
{
private string strClass; private string strAddress; //子类的构造函数
public Student ()
{
Console.WriteLine("我是子类的构造函数");
}
}

我们使用VS的单步调试,来看父类和子类显式构造函数的执行顺序,如下图(动态图片,可以看到过程):

很容易的可以发现,当创建子类对象的时候

①先调用了子类的构造函数

②调用了父类的构造函数

③执行了父类的构造函数

④执行了子类的构造函数

那么为什么会这样呢?

我尝试通过反编译看源码来解释这个原因,但是反编译的结果如下,

没有发现有什么特别的地方可以解释这个原因。

最后还是查阅微软的MSDN官方文档找到了答案(原文地址点击这里

根据微软官方的代码示例,那么下面的代码的效果也是相同的

//子类的构造函数
public Student ()
{
Console.WriteLine("我是子类的构造函数"); } //这里的代码和上面的代码效果是相同的
public Student()
:base()
{
Console.WriteLine("我是子类的构造函数");
}

也就是说只要在子类显式的声明了无参的构造函数,在实例化子类的对象是,子类的无参构造函数都会去调用父类无参的构造函数。

那么,如果父类没有这个无参的构造函数则会报错。

如下面的代码:

 class Person
{
private int nAge;
protected string strName;
double douHeight;
public string strEateType; //父类的构造函数
//public Person()
//{
// Console.WriteLine("我是父类的构造函数");
//}       //父类的有参数的构造函数,这里覆盖了无参的构造函数
public Person (string str)
{
Console.WriteLine("我是父类的构造函数{0}",str);
} public void Hello()
{
Console.WriteLine("我可以说Hello!");
}
public void Run()
{
Console.WriteLine("我可以跑!");
}
} class Student : Person
{
private string strClass; private string strAddress; //子类的无参构造函数
public Student ()
{
Console.WriteLine("我是子类的构造函数"); } public Student(string strName)
{
Console.WriteLine("我的名字叫{0}",strName);
}
}

这时候编译会报错,

因为在父类中有参数的构造函数覆盖了无参数的构造函数,所以在子类的无参数的构造函数没办法回调父类的无参数的构造函数初始化父类的成员变量。所以报错。

那么在初始化子类的时候,为什么要调用父类的构造函数呢?

在初始化子类之前需要通过构造函数初始化父类的成员变量

父类的构造函数先于子类的构造函数执行的意义是什么呢?

当在父类的构造函数中和子类的构造函数中为父类的非私有成员变量赋不同默认值。当实例化子类,子类要调用构造函数初始化成员变量,如果先执行了子类的构造函数,再执行父类的构造函数,父类成员字段的值会覆盖子类成员字段的值。但是我们想得到的是子类的属性值。所以为了解决数据冲突,父类的构造函数要先于子类的构造函数执行。

如下面的代码:

class Person
{
private int nAge;
private string strName;
double douHeight;
public string strEateType; // 父类的构造函数
public Person()
{
//再父类中对strEateType赋初始值
this.strEateType = "吃饭";
Console.WriteLine("我是父类的构造函数{0}", strEateType);
}
} class Student : Person
{
private string strClass;
private string strAddress; //子类的构造函数
public Student()
{
//在子类中对strEateType赋初始值
this.strEateType = "吃面条";
Console.WriteLine("我是子类的构造函数{0}",strEateType); }
}

这时候我们通过,声明子类对象访问strEateType的值,如下:

Student stu1 = new Student();
//stu1.
string str = stu1.strEateType.ToString();
Console.WriteLine(str); Console.ReadKey();

这里肯定是要打印出子类的属性strEateType的值,如果先执行子类构造函数对strEateType赋值,然后父类的构造函数赋值覆盖strEateType的初始值。那么打印出的将是父类成员字段的值。所以,父类的构造函数先于子类的构造函数执行。

打印结果如下:

三,子类是否可以有和父类的同名方法

看下面的代码,我们声明一个父类Person:

 class Person
{
private int nAge;
private string strName;
double douHeight;
public string strEateType; public readonly string strrrr;
// 父类的构造函数
public Person()
{
this.strEateType = "吃饭";
Console.WriteLine("我是父类的构造函数{0}", strEateType);
} public Person(string str)
{
this.strName = str;
Console.WriteLine("我是父类的构造函数{0}", str);
} public void Hello()
{
Console.WriteLine("我可以说地球人的Hello!");
}
public void Run()
{
Console.WriteLine("我可以跑!");
}
}

声明一个子类继承Person,代码如下:

class Worker:Person
{
public void Hello()
{
Console.WriteLine("我是工人会说Hello!");
} public new void Run()
{
Console.WriteLine("我是工人我会奔跑!");
}
}

然后实例化Worker对象,打印Hello方法,结果如下图:

 

这是为什么呢?编译器已经告诉了我们,如下图:



看出来是子类的方法隐藏了父类的方法。

既然子类可以定义和父类同名的方法,那么是否可以定同名的字段呢?答案是肯定的,而且会像同名方法一样,子类同名字段会隐藏父类同名的字段。


毕业实习交流群:221376964。你也可以关注我的新浪微博进行交流。

版权声明:本文为博主原创文章,未经博主允许不得转载。

最新文章

  1. 使用c/c++扩展python
  2. 百度sdk定位不成功,关闭定位
  3. TF-IDF提取行业关键词
  4. Window 注册程序关联后缀文件,双击运行
  5. selenium—JS点击方法
  6. Dev的DocumentManager添加窗体
  7. Portion of class Throwable’s inheritance hierarchy
  8. LinkedIn高级分析师王益:大数据时代的理想主义和现实主义(图灵访谈)
  9. replace empty char with new string,unsafe method和native implementation的性能比较
  10. mac os 终端提示 you have new mail
  11. Foundation Sorting: Single List Insertion Sort
  12. QR码生成原理
  13. DNS 域名系统 (Domain Name System)
  14. 完美解决--用VS中的Git做代码管理器,与他人共享代码
  15. 做一个有产品思维的研发:Scrapy安装
  16. Matlab - 各种函数学习
  17. i春秋SRC部落携手同程SRC发布首届漏洞提交任务
  18. C# System.Collections.SortedList
  19. 通过SSH克隆远程仓库(GitLab)到本地
  20. 如何使用JDBC连接Mysql数据库

热门文章

  1. HDU 1009 FatMouse' Trade (贪心算法)
  2. movielens 时间戳是秒级别的
  3. hdu2680 choose the best route
  4. NOR Flash的学习
  5. ECG心电图数据1
  6. 自适应XAML布局经验总结 (二) 局部布局设计模式1
  7. 初探Angular_03 组件中模板数据绑定
  8. In file included from adlist.c:34:0: zmalloc.h:50:31: 致命错误:jemalloc/jemalloc.h:没有那个文件或目录
  9. C#使用PriorityQueue
  10. chrome 插件学习笔记(一)