前几天美学课上,老师说到形态美,便举例你看韩红,
这么多年没留过长发,正是因为她明白短发才是最适合自己的形态美。
第一排的男生大声的接了一句:“她可能是害怕留长发会被认成刘欢!”
看看基类、类、对角交互中使用Me,MyClass,MyBase的情况
一、Me
Me 关键字为引用代码正在运行的类或结构的特定实例提供了一条途径。
Me 的行为方式类似于引用当前实例的对象变量或结构变量。
在向另一个类、结构或模块中的过程传递关于某个类或结构的当前执行实例的信息时,使用 Me 尤其有用。
结论:Me可以省略,它不会破坏既有的规则,比如重载、重写、隐藏,它就是一个墙头草,每个规则来了它都会适应。
重写来了,它将遵循它的虚拟方法,按实际对象的类型来解释Me;
隐藏来了,它将遵循它的非虚拟方法,按变量的类型来解释Me;
它的作用仅仅就是少写几个字母。
先在Person类中添加一个ToString方法
Public Overrides Function ToString() As String '重写Object中的tostring Return Name End Function
注意:1、为什么在基本中用Overides?
因为.Net Framework中所有类最终都派生于System.Object。即使没有使用Inherits,它也是如此。
故Person类是由Object类继承而来,可用Overrides
2、默认情况下,ToString返回类的名称(即使没有定义)。这个重写成返回Name属性。
然后,我们改写主程序中的显示:
txtName.Text = .ToString
Public Overrides Function ToString() As String '重写Object中的tostring Return Me.Name '多态时发生变化 End Function
但是这个是不必要的,对类中所有方法调用来说,Me就是默认的。不仅对重写还是隐藏都是如此,所以上面显式Me不影响
通常,应省略Me,避免额外的输入。
下面看一下重写时,Me的情况:
添加OfficeEmployee类,增加属性OfficeNumber,并重写Name属性。
Public Class officeEmployee Inherits Employee Private mOffice As String Public Property OfficeNumber() As String Get Return mOffice End Get Set(value As String) mOffice = value End Set End Property Public Overloads Overrides Property Name() As String Get Return MyBase.Name(NameType.informal) '返回非正式中 End Get Set(value As String) MyBase.Name = value '基类存储值 End Set End Property End Class注意:上例中MyBase指的是基类,即Employee。(而不是Person)
主程序中代码如下:运行一下,可以看到显示的是Freddy
这是什么原因呢?
因为重写发生虚拟方法。直接调用方法或者通过Me关键来调用方法时:
会调用当前对象上的方法,所以这些方法调用会遵循和外部方法调用相同的规则,即,
如果Tostring方法被继承,其下的了类(Employee或OfficeEmployee)发生重写,这个Me不再是Person,而是
当前实际的对象类型(例中为OfficEmployee)。
所以上例是这样的:
OfficeEmployee中对象temp,调用tostring(这是从Person继承而来),于是使用Person的Tostring方法,该方法中的
的Me不再是Person,而是OfficEmployee,即调用的是Temp.Name,这样转到本类代码的Name属性,这个属性又
返回的是基类(Employee)的Name(nametype)属性,转到Employee代码中提取Name(NameType)属性,也就是非
正式名Freddy。
具体流程如下箭头所示:
下面再看一下Me在隐藏情况下的表现:
在OfficeEmployee类中,改变重写代码为隐藏,如下:
Public Shadows Property Name() As String Get Return MyBase.Name(NameType.informal) End Get Set(value As String) MyBase.Name = value End Set End Property
主程序中,变量与引用对象类型改变一下:
可以看到,这里它又遵循了隐藏的规则。
变量类型为Employee,在调用toString,将进入Person版本的ToString,里面用了Me.Name,这个Me就是当前的Employ,
所以Name值就是Employee.Name(注意这里不Person.Name,也不是OfficeEmployee.Name),所以值就是Fred。
结论:Me不会影响重写或隐藏的功能。同样也说明了Me是可以省略的。
二、MyBase
MyBase 关键字的行为方式类似于引用当前类实例的基类的对象变量。
MyBase 通常用于访问派生类中被重写或被隐藏的基类成员。
MyBase.New 用于从派生类构造函数中显式调用基类构造函数。
MyBase只能引用直接父类,无法继续沿继承链向上定位。
MyBase可以从父类中调用或使用任何Pulbic,Friend,Protected元素。
无论是重写还是隐藏,它都能突破这种限制,访问父类方法。
如果说Me是墙头草,那么MyBase就是定神针,不为外界影响,直接可以引用基类元素。
上例中,OfficeEmployee类中隐藏代码:
Public Shadows Property Name() As String Get Return MyBase.Name(NameType.informal) End Get Set(value As String) MyBase.Name = value End Set End Property使用了MyBase,无论世界如何改变,它始终就固定位置到了父类Employee中。使用的是Employee中的方法。
结论:突破重写或隐藏的封锁,用MyBase把基类的功能合并到子类中。
三、MyClass
MyClass 关键字的行为方式类似于最初实现时引用类的当前实例的对象变量。
MyClass 类似于 Me,但调用前者的所有方法都被视为是 NotOverridable。
MyClass固执地坚定自己,不会因外界的变化。
可以说MyClass与MyBase都是一类角色,都不会因外界的变化而变,MyClass只指向本类,MyBase只指向父类。
看一下变化:
改变Person中ToString的方法,用MyClass,那么它始终用的是Person版本Name。
Public Overrides Function ToString() As String '重写Object中的tostring Return MyClass.Name '强制使用本类版本 End Function
为什么运行起来其值为空呢?
因为上面主程序中,无论怎么赋值,Person版本中的Name值一直没有赋值,它一直是空串。
所以,尽管用的是Temp.Tostring,它将调用Person中的ToSTring方法,而这个方法调用的是MyClass.Name,即调用的是
Person.Name.
四、My
My 功能提供了容易而直观的方法来访问大量 .NET Framework 类,
从而使 Visual Basic 用户能够与计算机、应用程序、设置、资源等进行交互。
可以把它看作是“系统”功能的调用。
结论:在类中,MyClass,MyBase起到不可替代的作用。