目录:

【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。你也可以关注我的新浪微博进行交流。

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

【C#小知识】C#中一些易混淆概念总结(五)---------继承 分类: C# 2014-02-06 22:05 1106人阅读 评论(0) 收藏的更多相关文章

  1. VS2010中使用命令行参数 分类: c/c++ 2014-07-11 22:24 634人阅读 评论(0) 收藏

    在Linux下编程习惯了使用命令行参数,故使用VS2010时也尝试了一下. 新建项目,c++编写程序如下: #include<iostream> #include<fstream&g ...

  2. jdbc知识问答 分类: 面试 2015-07-10 22:05 5人阅读 评论(0) 收藏

    1 JDBC连接数据库6步 Load the JDBC Driver Establish the Database Connection Create a Statement Object Execu ...

  3. iOS中UITextField 使用全面解析 分类: ios技术 2015-04-10 14:37 153人阅读 评论(0) 收藏

    //初始化textfield并设置位置及大小   UITextField *text = [[UITextField alloc]initWithFrame:CGRectMake(20, 20, 13 ...

  4. 解决ORA-29857:表空间中存在域索引和/或次级对象 & ORA-01940:无法删除当前连接的用户问题 分类: oracle sde 2015-07-30 20:13 8人阅读 评论(0) 收藏

    今天ArcGIS的SDE发生了一点小故障,导致系统表丢失,所以需要重建一下SDE数据库,在删除SDE用户和所在的表空间过程中遇到下面两个ORA错误,解决方法如下: 1)删除表空间时报错:ORA-298 ...

  5. centos中的配置文件 分类: B3_LINUX 2015-04-03 22:21 184人阅读 评论(0) 收藏

    /etc/profile:此文件为系统的每个用户设置环境信息,当用户第一次登录时,该文件被执行.并从/etc/profile.d目录的配置文件中搜集shell的设置. /etc/bashrc:为每一个 ...

  6. C/C++中const的用法 分类: C/C++ 2015-07-05 00:43 85人阅读 评论(0) 收藏

    const是C语言的关键字,经C++进行扩充,变得功能强大,用法复杂.const用于定义一个常变量(只读变量),当const与指针,引用,函数等结合起来使用时,情况会变得复杂的多.下面将从五个方面总结 ...

  7. 认识C++中的临时对象temporary object 分类: C/C++ 2015-05-11 23:20 137人阅读 评论(0) 收藏

    C++中临时对象又称无名对象.临时对象主要出现在如下场景. 1.建立一个没有命名的非堆(non-heap)对象,也就是无名对象时,会产生临时对象. Integer inte= Integer(5); ...

  8. Java获取项目中的路径 分类: Java Game 2014-08-14 10:17 122人阅读 评论(0) 收藏

    在项目中经常需要获取某个文件的路径: 在这里提供一些获取路径的方法.. 1.此种方式获取的路径,是当前类所在的路径: UserDAOTest.class.getResource("UserD ...

  9. Struts知识问答 分类: 面试 2015-07-10 22:01 4人阅读 评论(0) 收藏

    1. 简述Struts框架的初始化流程. 答案: 对于采用Struts框架的Web应用,在Web应用启动时就会加载并初始化控制器ActionServlet ActionServlet从struts-c ...

随机推荐

  1. linxu ssh 双端认证 不成功之authorized_keys

    linxu ssh 双端认证 不成功之authorized_keys liunx双端认证可以让我们更简便的在两台服务器之间传输文件,配置暂且不说,网上有大部分的文章可以搜索到,今天我要说的是在不成功的 ...

  2. 记一次项目使用webuploader爬坑之旅

       因前端页面开发使用的为VUE开发,又要支持IE9,遂只有基于webuploader封装一个上传组件.地址:https://github.com/z719725611/vue-upload-web ...

  3. 如何将word中的图片和文字导入自己的博客中

    目前大部分的博客作者在用Word写博客这件事情上都会遇到以下3个痛点: 1.所有博客平台关闭了文档发布接口,用户无法使用Word,Windows Live Writer等工具来发布博客.使用Word写 ...

  4. 【翻译】JavaScript框架的最终指南

    翻译原文链接 我的翻译小站 紧跟JavaScript框架的脚步是一个挑战.现在有太多的框架,几乎一个月就会出来一个新的.那么如何知道到底哪一个比较合适你的项目呢?它们分别有什么优点和缺点呢?你要如何开 ...

  5. c++ 内联函数 (讲解的TM真好)

    1.  内联函数 在C++中我们通常定义以下函数来求两个整数的最大值: 复制代码 代码如下: int max(int a, int b) {  return a > b ? a : b; } 为 ...

  6. linux信号量(转载)

    本文转载自http://blog.csdn.net/qinxiongxu/article/details/7830537 信号量一.什么是信号量信号量的使用主要是用来保护共享资源,使得资源在一个时刻只 ...

  7. codevs 1083

    这道题是看了人家大牛的解题报告: 对了,要说明一下,(A+B)&1 ,表示,判断(A+B)是奇数否? 下面给出代码: #include<iostream> #include< ...

  8. D3 drag

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. StructuredStream StateStore机制

    ref: https://jaceklaskowski.gitbooks.io/spark-structured-streaming/ StruncturedStream的statefule实现基于S ...

  10. 启用Nginx目录浏览功能的方法

    location / {           root /data/www/file                     //指定实际目录绝对路径:           autoindex on; ...