(类) 与 (结构) 类似,让我们可以定义并封装成一组相关项的数据类型。比如封装成结构,那么这个封装包的数据类型就为值类型;如封装成类,那么这个封装包的数据类型就为引用类型。

然而与结构的一个重要区别就是:类可以继承和扩展其他类的特性。

作为新类的基础的类称为“基类”;从基类派生出来的类称为“派生类”。派生类继承基类中定义的所有字段、属性、方法 和事件。我们只要对类进行一次开发和调试,就可以将其作为其他类的基类重复使用。

我们看一段代码:

Module Module1
Public Class baseClass
Private x As Integer '类的字段,建议使用字段私有化
Protected y As Integer '受保护的字段,只有派生类可调用
Public Sub New() '基类的构造函数
x = 10
Console.WriteLine("(1)、基类的 New 构造函数被调用")
End Sub
Public Overridable Function GetY(ByVal intValue As Integer) As Integer '基类的可被派生类重写的方法
GetY = Me.x * intValue
End Function
Protected Overrides Sub finalize() '基类的析构函数
Console.WriteLine("(8)、基类的析构函数 Finalize 被调用")
MyBase.Finalize()
End Sub Public Sub vbCrLfA() '基类的公共方法
Console.WriteLine(vbCrLf)
End Sub
End Class
End Module

这就是基类,要想一个类具有基类的特性,那就必须具有可被重写或可被重载的函数或方法,最好有 Protected 关键字命名的字段、属性方法,因为它标志着只有派生类可访问。有了这些条件,那此类才称得上可作为基类被继承。

下面我们来写一个由上面的类派生出来的另一个类:

Private Class derivedClass
Inherits baseClass
Public x As Single = 100.6273
Public Sub New()
MyBase.New() 'MyBase关键字表示引用当前类的基类的指定方法,这里为基类的New构造函数
Console.WriteLine("(2)、派生类的 New 构造函数被调用,引用了基类的构造函数")
End Sub
Protected Overrides Sub Finalize()
Console.WriteLine("(7)、派生类的析构函数 Finalize 被调用")
MyBase.finalize() '这里使用基类的析构函数类执行清理任务
End Sub
Public Overrides Function GetY(intValue As Integer) As Integer '这里重写了基类中 GetY 方法,但却不是用基类此方法的原始计算方法,而是进行了扩展和修改与完善
For i = 0 To intValue
Me.y += MyBase.GetY(i) '用基类的GetY方法获取到的Y值来进行累加的结果,基类GetY方法的参数为循环值
Next
Return Me.y '基类中用 protected 声明的 y 字段可以在派生类中使用
End Function
Public Sub intValue() '派生类的公共方法,对派生类中的X字段值取整
Int(x) '取整
End Sub
End Class

我们通过声明 Inherits 关键字,来标识该类的基类是谁,如果没有标识的类,比如 BaseHenry,VB.Net就会视其为派生自Object基类,这是种隐式继承,和结构是隐式继承自 ValueType 类的方式是一样的。

那这些overrides与 overload标识是什么意思呢?

我们使用继承,就要允许派生类中的某些属性或方法有与基类不同的行为,可以让派生类重写基类的方法或函数。但基类也得要同意你才能重写 才行啊,否则不就会发生命名冲突了吗?

因为派生类的对象是可以使用基类的公用成员的,那怎么知道哪些基类的方法与属性是被派生类重写了呢?所以就必须有一个约定:我们在基类中用 Overridable 修饰符来标识允许基类中的属性或方法在其派生类中可以被重写,没有标识的其实就是默认为有 NotOverridable 修饰符来隐式地标识,用于提醒编译器该属性或方法不可被重写;然后在派生类重写时,我们就用 Overrides 修饰符来标识哪些是重写自基类中定义的 Overridable 属性或方法 。

现在我们,调用派生类和基类的函数、字段和方法,来看看结果:

Sub Main()
Dim myValue As New derivedClass
myValue.GetY(5) '调用派生类的 GetY 函数
Console.WriteLine("(3)、调用派生类的 GetY 函数计算后返回 y 的值为:{0} ", myValue.GetY(5))
Dim x As Single = myValue.x '读取派生类中的公共字段变量的值
Console.WriteLine("(4)、调用派生类中 x 字段的值为:{0} ", x)
myValue.intValue() '调用派生类的公共方法
Dim b As Integer = myValue.x
Console.WriteLine("(5)、调用派生类的 intValue方法后派生类中 x 字段的值为:{0} ", b)
myValue.vbCrLfA() '调用基类的公共方法,因为 derivedClass 是 baseClass 类派生出的,所以可继承基类的所以公共成员
Console.WriteLine("(6)、调用基类的 vbCrLfA 公共方法进行换行,因为派生类可继承基类的所以公共成员")
Console.Read()
End Sub

结果如图:

注意:Finalize 方法只有当程序结束后才会被触发。
大家会奇怪,只在 Main 中实例化了一个派生类的对象,为什么先会出现执行“基类的构造”?

其实是派生类 derivedClass 中的这段代码在作怪:

Public Sub New()
MyBase.New() 'MyBase关键字表示引用当前类的基类的指定方法,这里为基类的New构造函数
Console.WriteLine("(2)、派生类的 New 构造函数被调用,引用了基类的 New 构造函数")
End Sub

我们必须要注意这一点,就是派生类的构造函数与析构函数都必须重新编写。New 构造函数一般用于打开文件、连接到数据库 、初始化变量以及处理任何需要在可使用对象前完成的其他任务。我们必须在 Sub New 构造函数中的第一行代码使用语句 MyBase.New(),来调用该类的基类的构造函数,以获得基类的性质。析构函数则是在 Sub Finalize 中执行完对派生类的的清理任务,如保存状态信息、关闭文件和与数据库的连接,以及执行在 释放对象前必须完成的其他任务之后,在析构函数的最后一句使用语句 MyBase.Finalize() 显式调用其基类的 Sub Finalize 方法,以析构MyBase.New ()构造的内容。所以你从程序运行结果中也可以很清楚的看出这一顺序。

讲到类的继承,我们还得明白 重载 与 隐藏 的问题。

类 的继承性(Inherits)与 重写(Overrides)的更多相关文章

  1. JAVASE(九)面向对象特性之 : 继承性、方法重写、关键字super、

    个人博客网:https://wushaopei.github.io/    (你想要这里多有) 1.继承性 1.1 为什么要类的继承性?(继承性的好处) ①减少了代码的冗余,提高了代码的复用性:②更好 ...

  2. Java类的设计----方法的重写、覆盖

    方法的重写.覆盖 在子类中可以根据需要对从父类中继承来的方法进行改造—覆盖方法(方法的重置.重写),在程序执行时,子类的方法将覆盖父类的方法. 覆盖方法必须和被覆盖方法具有相同的方法名称.参数列表和返 ...

  3. swift新建类和子类方法的重写与重载

    1.首先什么是重写与重载? 一.重写(override) override是重写(覆盖)了一个方法,以实现不同的功能.一般是用于子类在继承父类时,重写(重新实现)父类中的方法. 重写(覆盖)的规则: ...

  4. Python_类的继承与方法重写

    1.新建子类时,括号内要传入继承的父类名 2.super()方法:自动寻找当前类的父类,并调用父类的构造函数,初始化属性值 class Cup: #构造函数,初始化属性值 def __init__(s ...

  5. Java Object类中toString方法的重写

    Object类中的tostring方法的: 当我们输出一个对象时,实际是输出的是这个类中的tostring方法,是一个地址值,而不是类中的属性. 1 一:子类没有重写Object类中的toStrinn ...

  6. 两个类似的ViewModel一个可以重写事件,另一个不能重写事件,是哪里出了错。

    答:继承错了,BaseViewModel里面是事件.

  7. 【转载】 C++多继承中重写不同基类中相同原型的虚函数

    本篇随笔为转载,原文地址:C++多继承中重写不同基类中相同原型的虚函数. 在C++多继承体系当中,在派生类中可以重写不同基类中的虚函数.下面就是一个例子: class CBaseA { public: ...

  8. android开发之重写Application类

    在android应用开发中,重写Application也算是比较常见的,以前开发的一些程序太过于简单,都不要重写这个类,但是在真正的商业开发中,重写Application类几乎是必做的. 为什么要重写 ...

  9. Android_ 重写系统Crash处理类,保存Crash信息到SD卡 和 完美退出程序的方法

    转载时注明地址:http://blog.csdn.net/xiaanming/article/details/9344703 我们开发Android应用的时候,当出现Crash的时候,系统弹出一个警告 ...

随机推荐

  1. myeclipse8.6 for spring环境配置

     

  2. 0118——UIButtton

    1.Button的定义 UIButton *btn = [UIButton buttonWithType:UIButtonTypeRoundedRect]; Button有六种类型 enum { UI ...

  3. BaceModel

    https://github.com/nicklockwood/BaseModel 字典封装成model 自动封装 要求属性的名字与字典一样 不能有对象 如果其中有需要自己封装的对象属性 重写setW ...

  4. struts2的工作机制

    struts2的工作机制 原文:http://eoasis.iteye.com/blog/642586 概述 本章讲述Struts2的工作原理. 读者如果曾经学习过Struts1.x或者有过Strut ...

  5. I - u Calculate e

    Description A simple mathematical formula for e is where n is allowed to go to infinity. This can ac ...

  6. (转) c++ 迭代器

    原地址:http://www.cnblogs.com/marchtea/archive/2012/02/27/2370068.html 前言: 以下的内容为我阅读c++沉思录18,19,20章的笔记以 ...

  7. activiti笔记二:用户任务

    1, assignee 代替humanPerformer  功能 2, cadidateUsers代替potentialOwner功能 3, candidateGroups代替potentialOwn ...

  8. 高手总结的CSS执行顺序及其优先权问题汇总

    今天在看一本书时又看到了”CSS优 先权“这个问题,感觉这个问题还是比较重要的,也算是样式的特异性吧,尤其是在面对较多.较深层.较复杂的样式属性时,理解CSS的加权计算方法对于重写 样式属性之类的问题 ...

  9. 前端开发者使用JS框架的三个等级

    目前前端开发者使用JS框架是种很普遍的现象,因为框架可以加快开发速度,同时避免各类浏览器的兼容性问题.不过同样是用框架开发,不同开发者的境界水平还是有一定差距,本文将这些前端开发者分为三个等级. 第一 ...

  10. SSE2 Intrinsics各函数介绍[转]

    SIMD相关头文件包括: //#include <ivec.h>//MMX //#include <fvec.h>//SSE(also include ivec.h) //#i ...