Cthie和base(二)

Cthie和base(二)new 关键字引起了大家的不少关注 尤其感谢 AndersLiu 的补充 让我感觉博客园赋予的交流平台真的无所不在

new关键字引起了大家的不少关注,尤其感谢 Anders Liu的补充,让我感觉博客园赋予的交流平台真的无所不在。所以,我们就有必要继续这个话题,把我认为最值得关注的关键字开展下去,本文的重点是访问关键字(Access Keywords):base和this。虽然访问关键字不是很难理解的话题,我们还是有可以深入讨论的地方来理清思路。还是老办法,我的问题先列出来,您是否做好了准备。

  • 是否可以在静态方法中使用base和this,为什么?

  • base常用于哪些方面?this常用于哪些方面?

  • 可以base访问基类的一切成员吗?

  • 如果有三层或者更多继承,那么最下级派生类的base指向那一层呢?例如.NET体系中,如果以base访问,则应该是直接父类实例呢,还是最高层类实例呢?

  • 以base和this应用于构造函数时,继承类对象实例化的执行顺序如何?

  2. 基本概念

  base和this在C#中被归于访问关键字,顾名思义,就是用于实现继承机制的访问操作,来满足对对象成员的访问,从而为多态机制提供更加灵活的处理方式。 

  2.1 base关键字

  其用于在派生类中实现对基类公有或者受保护成员的访问,但是只局限在构造函数、实例方法和实例属性访问器中,MSDN中小结的具体功能包括:

  • 调用基类上已被其他方法重写的方法。

  • 指定创建派生类实例时应调用的基类构造函数。

  2.2 this关键字

  其用于引用类的当前实例,也包括继承而来的方法,通常可以隐藏this,MSDN中的小结功能主要包括:

  • 限定被相似的名称隐藏的成员

  • 将对象作为参数传递到其他方法

  • 声明索引器  

  3. 深入浅出

  3.1 示例为上

  下面以一个小示例来综合的说明,base和this在访问操作中的应用,从而对其有个概要了解,更详细的规则和深入我们接着阐述。本示例没有完全的设计概念,主要用来阐述base和this关键字的使用要点和难点阐述,具体的如下:

base和this示例
using System;
namespace Anytao.net.My_Must_net
{
public class Action
{
public static void ToRun(Vehicle vehicle)
{
Console.WriteLine("{0} is running.", vehicle.ToString());
}
}
public class Vehicle
{
private string name;
private int speed;
private string[] array = new string[10];


public Vehicle()
{
}
//限定被相似的名称隐藏的成员
        public Vehicle(string name, int speed)
{
this.name = name;
this.speed = speed;
}
public virtual  void ShowResult()
{
Console.WriteLine("The top speed of {0} is {1}.", name, speed);
}
public void Run()
{
//传递当前实例参数
            Action.ToRun(this);
}
//声明索引器,必须为this,这样就可以像数组一样来索引对象
        public string  this[int param]
{
get{return array[param];}
set{array[param] = value;}
}
}
public class Car: Vehicle
{
//派生类和基类通信,以base实现,基类首先被调用
//指定创建派生类实例时应调用的基类构造函数
        public Car()
: base("Car", 200)
{ }


public Car(string name, int speed)
: this()
{ }


public override void ShowResult()
{
//调用基类上已被其他方法重写的方法
            base.ShowResult();
Console.WriteLine("It's a car's result.");
}
}
public class Audi : Car
{
public Audi()
: base("Audi", 300)
{ }


public Audi(string name, int speed)
: this()

}
public override void ShowResult()
{
//由三层继承可以看出,base只能继承其直接基类成员
            base.ShowResult();
base.Run();
Console.WriteLine("It's audi's result.");
}
}
public class BaseThisTester
{
public static void Main(string[] args)
{
Audi audi = new Audi();
audi[1] = "A6";
audi[2] = "A8";
Console.WriteLine(audi[1]);
audi.Run();
audi.ShowResult();
}
}
}
 
 

 

  3.3 深入剖析 

  如果有三次或者更多继承,那么最下级派生类的base指向那一层呢?例如.NET体系中,如果以base访问,则应该是直接父类实例呢,还是最高层类实例呢?

  首先我们有必要了解类创建过程中的实例化顺序,才能进一步了解base机制的详细执行过程。一般来说,实例化过程首先要先实例化其基类,并且依此类推,一直到实例化System.Object为止。因此,类实例化,总是从调用System.Object.Object()开始。因此示例中的类Audi的实例化过程大概可以小结为以下顺序执行,详细可以参考示例代码分析。

  • 执行System.Object.Object();

  • 执行Vehicle.Vehicle(string name, int speed);

  • 执行Car.Car();

  • 执行Car.Car(string name, int speed);

  • 执行Audi.Audi();

  • 执行Audi.Audi(string name, int speed)。

  我们在充分了解其实例化顺序的基础上就可以顺利的把握base和this在作用于构造函数时的执行情况,并进一步了解其基本功能细节。

  下面更重要的分析则是,以ILDASM.exe工具为基础来分析IL反编译代码,以便更深层次的了解执行在base和this背后的应用实质,只有这样我们才能说对技术有了基本的剖析。

Main方法的执行情况为:

IL分析base和this执行

.method public hidebysig static void Main(string[] args) cil managed { .entrypoint // 代码大小 61 (0x3d)  .maxstack 3 .locals init (class Anytao.net.My_Must_net.Audi V_0) IL_0000: nop //使用newobj指令创建新的对象,并调用构造函数初始化  IL_0001: newobj instance void Anytao.net.My_Must_net.Audi::.ctor() IL_0006: stloc.0 IL_0007: ldloc.0 IL_0008: ldc.i4.1 IL_0009: ldstr "A6" IL_000e: callvirt instance void Anytao.net.My_Must_net.Vehicle::set_Item(int32, string) IL_0013: nop IL_0014: ldloc.0 IL_0015: ldc.i4.2 IL_0016: ldstr "A8" IL_001b: callvirt instance void Anytao.net.My_Must_net.Vehicle::set_Item(int32, string) IL_0020: nop IL_0021: ldloc.0 IL_0022: ldc.i4.1 IL_0023: callvirt instance string Anytao.net.My_Must_net.Vehicle::get_Item(int32) IL_0028: call void [mscorlib]System.Console::WriteLine(string) IL_002d: nop IL_002e: ldloc.0 IL_002f: callvirt instance void Anytao.net.My_Must_net.Vehicle::Run() IL_0034: nop IL_0035: ldloc.0 //base.ShowResult最终调用的是最高级父类Vehicle的方法, //而不是直接父类Car.ShowResult()方法,这是应该关注的  IL_0036: callvirt instance void Anytao.net.My_Must_net.Vehicle::ShowResult() IL_003b: nop IL_003c: ret } // end of method BaseThisTester::Main 

因此,对重写父类方法,最终指向了最高级父类的方法成员。

 

  4. 通用规则

  • 尽量少用或者不用base和this。除了决议子类的名称冲突和在一个构造函数中调用其他的构造函数之外,base和this的使用容易引起不必要的结果。

  • 在静态成员中使用base和this都是不允许的。原因是,base和this访问的都是类的实例,也就是对象,而静态成员只能由类来访问,不能由对象来访问。

  • base是为了实现多态而设计的。

  • 使用this或base关键字只能指定一个构造函数,也就是说不可同时将this和base作用在一个构造函数上。

  • 简单的来说,base用于在派生类中访问重写的基类成员;而this用于访问本类的成员,当然也包括继承而来公有和保护成员。

  • 除了base,访问基类成员的另外一种方式是:显示的类型转换来实现。只是该方法不能为静态方法。 

编程小号
上一篇 2025-01-09 15:51
下一篇 2025-01-09 15:40

相关推荐

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
如需转载请保留出处:https://bianchenghao.cn/bian-cheng-ji-chu/106296.html