DotNet面试题解析04-类型、方法与继承

类型基础

类型Type简述

.NET 中主要的类型就是值类型和引用类型,所有类型的基类就是 System.Object ,也就是说我们使用 FCL 提供的各种类型的、自定义的所有类型都最终派生自 System.Object ,因此他们也都继承了 System.Object 提供的基本方法。

typesystem.gif

在 .NET 代码中,我们可以很方便的创建各种类型,一个简单的数据模型、复杂的聚合对象类型、或是对客观世界实体的抽象。类 (class) 是最基础的 C# 类型,支持继承与多态。
一个 c# 类 Class 主要包含两种基本成员:

  • 状态(字段、常量、属性等)
  • 操作(方法、事件、索引器、构造函数等)

利用创建的类型(或者系统提供的),可以很容易的创建对象的实例。使用 new 运算符创建,该运算符为新的实例分配内存,调用构造函数初始化该实例,并返回对该实例的引用,如下面的语法形式:

1
<类名>  <实例名> = new <类名>([构造函数的参数])

创建后的实例对象,是一个存储在内存上(在线程栈或托管堆上)的一个对象,那可以创造实例的类型在内存中又是一个什么样的存在呢?她就是 类型对象(Type Object)

类型对象(Type Object)

1
2
3
4
5
6
int a = 123;                                                           // 创建int类型实例a
int b = 20; // 创建int类型实例b
var atype = a.GetType(); // 获取对象实例a的类型Type
var btype = b.GetType(); // 获取对象实例b的类型Type
Console.WriteLine(System.Object.Equals(atype,btype)); //输出:True
Console.WriteLine(System.Object.ReferenceEquals(atype, btype)); //输出:True

任何对象都有一个 GetType() 方法(基类 System.Object 提供的),该方法返回一个对象的类型,类型上面包含了对象内部的详细信息,如字段、属性、方法、基类、事件等等(通过反射可以获取)。在上面的代码中两个不同的 int 变量的类型( int.GetType() )是同一个 Type ,说明 int 在内存中有唯一一个(类似静态的) Systen.Int32 类型。

上面获取到的 Type 对象( Systen.Int32 )就是一个类型对象,她同其他引用类型一样,也是一个引用对象,这个对象中存储了 int32 类型的所有信息(类型的所有元数据信息)。

关于类型对象(Object Type):

  • 每一个类型(如 System.Int32 )在内存中都会有一个唯一的类型对象,通过(int)a.GetType() 可以获取该对象;
  • 类型对象( Object Type )存储在内存中一个独立的区域,叫 加载堆(Load Heap),加载堆是在进程创建的时候创建的,不受 GC 垃圾回收管制,因此类型对象一经创建就不会被释放的,他的生命周期从 AppDomain 创建到结束;
  • 每个引用对象都包含两个附加成员:TypeHandle 和 同步块索引,其中 TypeHandle 就指向该对象对应的类型对象;
  • 类型对象的加载由 class loader 负责,在第一次使用前加载;
  • 类型中的静态字段就是存储在这里的(加载堆上的类型对象),所以说静态字段是全局的,而且不会释放;

可以参考下面的图,第一幅图描述了对象在内存中的一个关系, 第二幅图更复杂,更准确、全面的描述了内存的结构分布。

Anytao_15_3.jpg

image_thumb_1.png

方法表

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public void DoTest()
{
B1 b1 = new B1(); B2 b2 = new B2();
b1.Print(); b2.Print(); //按预期应该输出 B1、B2

A ab1 = new B1(); A ab2 = new B2();
ab1.Print(); ab2.Print(); //这里应该B1,A
}

public class A
{
public virtual void Print() { Console.WriteLine("A"); }
}
public class B1 : A
{
public override void Print() { Console.WriteLine("B1"); }
}
public class B2 : A
{
public new void Print() { Console.WriteLine("B2"); }
}

方法表的加载:

  • 方法表的加载时 父类在前,子类在后,首先加载的是固定的4个来自 System.Object 的虚方法:ToString, Equals, GetHashCode, Finalize;
  • 然后加载父类 A 的虚方法;
  • 加载自己的方法;
  • 最后是构造方法:静态构造函数 .cctor() ,对象构造函数 .ctor() ;

方法的调用:
当执行代码 b1.Print() 时(此处只关注方法调用,忽略方法的继承等因素),通过 b1 的 TypeHandel 找到对应类型对象,然后找到 方法表槽,然后是对应的 IL 代码,第一次执行的时候,JIT 编译器需要把 IL 代码编译为本地机器码,第一次执行完成后机器码会保留,下一次执行就不需要 JIT 编译了。这也是为什么说 .NET 程序启动需要预热的原因。

.NET中的继承本质

方法表的创建过程是从父类到子类自上而下的,这是 .NET 中继承的很好体现,当发现有覆写父类虚方法会覆盖同名的父方法,所有类型的加载都会递归到 System.Object 类。

  • 继承是可传递的,子类是对父类的扩展,必须继承父类方法,同时可以添加新方法。
  • 子类可以调用父类方法和字段,而父类不能调用子类方法和字段。
  • 子类不光继承父类的公有成员,也继承了私有成员,只是不可直接访问。
  • new 关键字在虚方法继承中的阻断作用,中断某一虚方法的继承传递。

用基类(A)和用本身B1声明到底有什么区别呢?

1
2
B1 b1 = new B1();
A ab1 = new B1();

无论用什么做引用声明,哪怕是 object,等号右边的[ = new 类型()]都是没有区别的,也就说说对象的创建不受影响的,b1 和 ab1 对象在内存结构上是一致的;
他们的的差别就在引用指针的类型不同,这种不同在编码中智能提示就直观的反应出来了,在实际方法调用上也与引用指针类型有直接关系;
综合来说,不同引用指针类型对于对象的创建(new操作)不影响;但对于对象的使用(如方法调用)有影响,这一点在上面代码的执行结果中体现出来了!

上面调用的IL代码:

151257-20160306225941018-923782671.png

对于虚方法的调用,在 IL 中都是使用指令 callvirt,该指令主要意思就是具体的方法在运行时动态确定的:

callvirt 使用虚拟调度,也就是根据引用类型的动态类型来调度方法,
callvirt 指令根据引用变量指向的对象类型来调用方法,在运行时动态绑定,主要用于调用虚方法。

不同的类型指针在虚拟方法表中有不同的附加信息作为标志,来区别其访问的地址区域,称为 offset。不同类型的指针只能在其特定地址区域内执行。

编译器在方法调用时还有一个原则:

执行就近原则:对于同名字段或者方法,编译器是按照其顺序查找来引用的,也就是首先访问离它创建最近的字段或者方法。

151257-20160306225942315-1520940493.png

在 C# 中,new 关键字可用作运算符、修饰符或约束。

  • 1,new 运算符:用于创建对象和调用构造函数。
  • 2,new 修饰符:在用作修饰符时,new 关键字可以显式隐藏从基类继承的成员。
  • 3,new 约束:用于在泛型声明中约束可能用作类型参数的参数的类型。

.NET中的继承

抽象类

抽象类提供多个派生类共享基类的公共定义,它既可以提供抽象方法,也可以提供非抽象方法。抽象类不能实例化,必须通过继承由派生类实现其抽象方法,因此对抽象类不能使用 new 关键字,也不能被密封。

基本特点:

  • 抽象类使用 Abstract 声明,抽象方法也是用 Abstract 标示;
  • 抽象类不能被实例化;
  • 抽象方法必须定义在抽象类中;
  • 抽象类可以继承一个抽象类;
  • 抽象类不能被密封(不能使用 sealed);
  • 同类 Class 一样,只支持单继承;
1
2
3
4
5
public abstract class AbstractUser
{
public int Age { get; set; }
public abstract void SetName(string name);
}

151257-20160306225944377-748332278.png

接口

接口简单理解就是一种规范、契约,使得实现接口的类或结构在形式上保持一致。实现接口的类或结构必须实现接口定义中所有接口成员,以及该接口从其他接口中继承的所有接口成员。

基本特点:

  • 接口使用 interface 声明;
  • 接口类似于抽象基类,不能直接实例化接口;
  • 接口中的方法都是抽象方法,不能有实现代码,实现接口的任何非抽象类型都必须实现接口的所有成员:
  • 接口成员是自动公开的,且不能包含任何访问修饰符。
  • 接口自身可从多个接口继承,类和结构可继承多个接口,但接口不能继承类。
1
2
3
4
5
public interface IUser
{
int Age { get; set; }
void SetName(string name);
}

下面是 IUser 接口定义的 IL 代码,看上去是不是和上面的抽象类 AbstractUser 的 IL 代码差不多!接口也是使用.Class ~ abstract 标记,方法定义同抽象类中的方法一样使用 abstract virtual 标记。因此可以把接口看做是一种特殊的抽象类,该类只提供定义,没有实现。

151257-20160306225945627-434523504.png

另外一个小细节,上面说到接口是一个特殊的类型,不继承 System.Object,通过 IL 代码其实可以证实这一点。无论是自定义的任何类型还是抽象类,都会隐式继承 System.Object,AbstractUser 的 IL 代码中就有 “extends [mscorlib]System.Object”,而接口的 IL 代码并没有这一段代码。

继承

在.NET中继承的主要两种方式就是类继承和接口继承,两者的主要思想是不一样的:

  • 类继承强调父子关系,是一个“IS A”的关系,因此只能单继承(就像一个人只能有一个Father);
  • 接口继承强调的是一种规范、约束,是一个“CAN DO”的关系,支持多继承,是实现多态一种重要方式。

题目答案解析

1. 所有类型都继承System.Object吗?

基本上是的,所有值类型和引用类型都继承自 System.Object,接口是一个特殊的类型,不继承自 System.Object。

2. 解释virtual、sealed、override和abstract的区别

  • virtual 申明虚方法的关键字,说明该方法可以被重写
  • sealed 说明该类不可被继承
  • override 重写基类的方法
  • abstract 申明抽象类和抽象方法的关键字,抽象方法不提供实现,由子类实现,抽象类不可实例化。

3. 接口和类有什么异同?

不同点:

1、接口不能直接实例化。

2、接口只包含方法或属性的声明,不包含方法的实现。

3、接口可以多继承,类只能单继承。

4、表达的含义不同,接口主要定义一种规范,统一调用方法,也就是规范类,约束类,类是方法功能的实现和集合

相同点:

1、接口、类和结构都可以从多个接口继承。

2、接口类似于抽象基类:继承接口的任何非抽象类型都必须实现接口的所有成员。

3、接口和类都可以包含事件、索引器、方法和属性。

4. 抽象类和接口有什么区别?

1、继承:接口支持多继承;抽象类不能实现多继承。

2、表达的概念:接口用于规范,更强调契约,抽象类用于共性,强调父子。抽象类是一类事物的高度聚合,那么对于继承抽象类的子类来说,对于抽象类来说,属于”Is A”的关系;而接口是定义行为规范,强调“Can Do”的关系,因此对于实现接口的子类来说,相对于接口来说,是”行为需要按照接口来完成”。

3、方法实现:对抽象类中的方法,即可以给出实现部分,也可以不给出;而接口的方法(抽象规则)都不能给出实现部分,接口中方法不能加修饰符。

4、子类重写:继承类对于两者所涉及方法的实现是不同的。继承类对于抽象类所定义的抽象方法,可以不用重写,也就是说,可以延用抽象类的方法;而对于接口类所定义的方法或者属性来说,在继承类中必须重写,给出相应的方法和属性实现。

5、新增方法的影响:在抽象类中,新增一个方法的话,继承类中可以不用作任何处理;而对于接口来说,则需要修改继承类,提供新定义的方法。

6、接口可以作用于值类型(枚举可以实现接口)和引用类型;抽象类只能作用于引用类型。

7、接口不能包含字段和已实现的方法,接口只包含方法、属性、索引器、事件的签名;抽象类可以定义字段、属性、包含有实现的方法。

5. 重载与覆盖的区别?

重载:当类包含两个名称相同但签名不同(方法名相同,参数列表不相同)的方法时发生方法重载。用方法重载来提供在语义上完成相同而功能不同的方法。

覆写:在类的继承中使用,通过覆写子类方法可以改变父类虚方法的实现。

主要区别

1、方法的覆盖是子类和父类之间的关系,是垂直关系;方法的重载是同一个类中方法之间的关系,是水平关系。
2、覆盖只能由一个方法,或只能由一对方法产生关系;方法的重载是多个方法之间的关系。
3、覆盖要求参数列表相同;重载要求参数列表不同。
4、覆盖关系中,调用那个方法体,是根据对象的类型来决定;重载关系,是根据调用时的实参表与形参表来选择方法体的。

6. 在继承中new和override相同点和区别?看下面的代码,有一个基类A,B1和B2都继承自A,并且使用不同的方式改变了父类方法Print()的行为。测试代码输出什么?为什么?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public void DoTest()
{
B1 b1 = new B1(); B2 b2 = new B2();
b1.Print(); b2.Print(); //按预期应该输出 B1、B2

A ab1 = new B1(); A ab2 = new B2();
ab1.Print(); ab2.Print(); //这里应该输出什么呢?输出B1、A
}
public class A
{
public virtual void Print() { Console.WriteLine("A"); }
}
public class B1 : A
{
public override void Print() { Console.WriteLine("B1"); }
}
public class B2 : A
{
// 显式隐藏从基类继承的成员,也就是说此方法被隐藏,被调用ab2.Print()时,输出基类中的方法
public new void Print() { Console.WriteLine("B2"); }
}

7. 下面代码中,变量a、b都是int类型,代码输出结果是什么?

1
2
3
4
5
6
int a = 123;
int b = 20;
var atype = a.GetType();
var btype = b.GetType();
Console.WriteLine(System.Object.Equals(atype,btype)); //输出True
Console.WriteLine(System.Object.ReferenceEquals(atype,btype)); //输出True

8.class中定义的静态字段是存储在内存中的哪个地方?为什么会说她不会被GC回收?

随类型对象存储在内存的加载堆上,因为加载堆不受GC管理,其生命周期随 AppDomain,不会被 GC 回收。

参考:

.NET面试题解析(04)-类型、方法与继承

2md