使用 abstract 关键字可以创建不完整且必须在派生类中实现的类和 class 成员。

使用 sealed 关键字可以防止继承以前标记为 virtual 的类或某些类成员。

abstract修饰符可以用于修饰类、方法、属性、索引器(indexer)和事件。

1、抽象类

抽象类具有以下功能:
抽象类不能实例化。
抽象类可能包含抽象方法和访问器。
无法使用 sealed 修饰符来修改抽象类,因为两个修饰符的含义相反。 sealed 修饰符阻止类被继承,而abstract 修饰符要求类被继承。
派生自抽象类的非抽象类,必须包含全部已继承的抽象方法和访问器的实际实现。

通过在类定义前面放置关键字 abstract,可以将类声明为抽象类。 例如:

public abstract class A
{
// Class members here.
}

抽象类不能实例化(但是可以定义变量)。 抽象类的用途是提供一个可供多个派生类共享的通用基类定义。 例如,类库可以定义一个抽象类,将其用作多个类库函数的参数,并要求使用该库的程序员通过创建派生类来提供自己的类实现。

Expression arrayAccessExpr = Expression.ArrayAccess()//错误的 抽象方法不能实例化
Expression arrayAccessExpr = Expression.ArrayAccess( arrayExpr,indexes);//正确的,作为基类 用来接收子类

一般情况下抽象类中可以包含抽象方法和非抽象方法,静态类中只能包含静态成员,所有的静态成员也只能包含在静态类中。

因为静态static的本质是包含abstract和sealed的,所以静态方法可以包含在抽象类中,并通过抽象类名.静态方法  来使用该方法。又因为静态包含sealed特性,所以不可以被继承。

2、抽象方法

抽象方法有以下特征:

1.抽象方法是隐式的虚拟方法。

2.抽象方法的声明只能在抽象类中。

3由于抽象方法声明不提供实际的实现,因此没有方法主体;方法声明仅以分号结尾,且签名后没有大括号({ })。实现由方法 override 提供,它是非抽象类的成员。

4.方法体的实现被覆写方法提供,覆写方法是一个非抽象类的成员。

5.抽象属性的行为和抽象方法相像,除了不同的声明形式。

6.在一个静态属性中使用abstract 是一个错误。

*一个抽象属性能够通过派生类使用 override 实现.

抽象类也可以定义抽象方法。 方法是将关键字 abstract 添加到方法的返回类型的前面。 例如:

public abstract class A
{
public abstract void DoWork(int i);
}

抽象方法没有实现,所以方法定义后面是分号,而不是常规的方法块。 抽象类的派生类必须实现所有抽象方法。 当抽象类从基类继承虚方法时,抽象类可以使用抽象方法重写该虚方法。 例如:

// compile with: -target:library
public class D
{
public virtual void DoWork(int i)
{
// Original implementation.
}
} public abstract class E : D
{
public abstract override void DoWork(int i);
} public class F : E
{
public override void DoWork(int i)
{
// New implementation.
}
}

如果将 virtual 方法声明为 abstract,则该方法对于从抽象类继承的所有类而言仍然是虚方法。 继承抽象方法的类无法访问方法的原始实现,因此在上一示例中,类 F 上的 DoWork 无法调用类 D 上的 DoWork。通过这种方式,抽象类可强制派生类向虚拟方法提供新的方法实现。

 抽象实现部分抽象方法 

但是通过声明派生类也为抽象,我们可以避免所有或特定的虚方法的实现,

abstract class MyAbs
{
public abstract void AbMethod1();
public abstract void AbMethod2();
}
//not necessary to implement all abstract methods
//partial implementation is possible
abstract class MyClass1 : MyAbs
{
public override void AbMethod1()
{
Console.WriteLine("Abstarct method #1");
}
}
class MyClass : MyClass1
{
public override void AbMethod2()
{
Console.WriteLine("Abstarct method #2");
}
}
class MyClient
{
public static void Main()
{
MyClass mc = new MyClass();
mc.AbMethod1();
mc.AbMethod2();
}
}

3、抽象类和接口

一个抽象类必须为所有的接口成员提供实现

一个抽象类也能从一个接口来实现,这种情况,我们必须为所有的方法提供方法体,这些方法是来自接口

interface IInterface
{
void Method1();
}
abstract class MyAbs : IInterface
{
public void Method1()
{
Console.WriteLine("Method implemented from the IInterface");
}
}
class MyClass : MyAbs//must implement base class abstract method
{
}
class MyClient
{
public static void Main()
{
MyClass mc = new MyClass();
mc.Method1();
}
}

一个用于实现接口的抽象类可能把接口方法安排到抽象方法上。例如

interface I
{
void M();
}
abstract class C: I
{
public abstract void M();
}

4、密封类和类成员

过在类定义前面放置关键字 sealed,可以将类声明为密封类。 例如:

public sealed class D
{
// Class members here.
}

密封类不能用作基类。 因此,它也不能是抽象类。 密封类禁止派生。 由于密封类从不用作基类,所以有些运行时优化可以略微提高密封类成员的调用速度。

在对基类的虚成员进行重写的派生类上,方法、索引器、属性或事件可以将该成员声明为密封成员。 在用于以后的派生类时,这将取消成员的虚效果。 方法是在类成员声明中将 sealed 关键字置于 override 关键字前面。 例如:

public class D : C
{
public sealed override void DoWork() { }
}

最新文章

  1. ThinkPhp 验证码不显示图片
  2. 使用 Swift 在 iOS 10 中集成 Siri —— SiriKit 教程
  3. 工作中积累整理-CSS样式表(一)
  4. make:cc 命令未找到的解决方法
  5. 关于Android中混淆的问题
  6. JSON的故事
  7. 微信5.4你所不知道的事 X5浏览引擎提速50%-80%
  8. Spark(2) - Developing Application with Spark
  9. winform自动添加同级目录下可执行文件的快捷方式到右键菜单中
  10. Oracle中的decode()函数
  11. ssh简单入门级案例教程
  12. 浅谈 Java JPDA
  13. mysql 之审计 init-connect+binlog完成审计功能
  14. 解决:win8.1 oepnvpn客户端 redirect-gateway def1无效,自动获取的IP没有网关问题
  15. jsp技术知识点
  16. BZOJ 1491 社交网络 Floyd 最短路的数目
  17. linux 定时执行任务 at atq atrm命令的使用
  18. UVa 12169 - Disgruntled Judge(拓展欧几里德)
  19. Problem C: 矩阵对角线求和
  20. ajax用json实现数据传输

热门文章

  1. Fastjson反序列化漏洞分析 1.2.22-1.2.24
  2. gin中的文件上传
  3. 字节跳动Web Infra发起 Modern.js 开源项目,打造现代 Web 工程体系
  4. Vue2和Vue3技术整理1 - 入门篇 - 更新完毕
  5. Linux 配置常用工具?
  6. ApacheCN 计算机视觉译文集 20210203 更新
  7. MySQL--数据表操作--行转列和列转行
  8. Lesson12——NumPy 字符串函数之 Part1:字符串操作函数
  9. 数据分析之客户价值模型(RFM)技术总结
  10. 垃圾陷阱 && [NOIP2014 提高组] 飞扬的小鸟