C# 多态性

转载自:MSDN

类似文章:点击打开链接

 Polymorphism(多态性)是一个希腊词,指“多种形态”,多态性具有两个截然不同的方面:

 发生此情况时,该对象的声明类型不再与运行时类型相同。

 在运行时,客户端代码调用该方法,CLR 查找对象的运行时类型,并调用虚方法的重写方法。 因此,你可以在源代码中调用基类的方法,但执行该方法的派生类版本。

派生类(子类)与基类(父类)之间强制转换不会丢失信息。

namespace PolymorphismTest{    public partial class Form1 : Form    {        public Form1()        {            InitializeComponent();        }        private void button1_Click(object sender, EventArgs e)        {            ParentClass parent = new ParentClass();            SubClass sub = new SubClass();            parent = ((ParentClass)sub);            //子类 父类 实例之间来回转换不会丢失信息           SubClass subNew = (SubClass)parent;        }    }    public class SubClass : ParentClass    {        public new string name = "SubClass";        public int age = 20;    }    public class ParentClass    {        string name = "ParentClass";     }}

执行到

SubClass subNew = (SubClass)parent

具体结果如下图:

C#  多态性

 例如,假定你有一个绘图应用程序,允许用户在绘图图面上创建各种形状。 你在编译时不知道用户将创建哪些特定类型的形状。 但应用程序必须跟踪创建的所有类型的形状,并且必须更新这些形状以响应用户鼠标操作。 你可以使用多态性通过两个基本步骤解决这一问题:

创建一个类层次结构,其中每个特定形状类均派生自一个公共基类。

使用虚方法通过对基类方法的单个调用来调用任何派生类上的相应方法。

 为 Shape 类提供一个名为 Draw 的虚方法,并在每个派生类中重写该方法以绘制该类表示的特定形状。创建一个 List 对象,并向该对象添加 Circle、Triangle 和 Rectangle。 若要更新绘图图面,请使用 foreach 循环对该列表进行循环访问,并对其中的每个 Shape 对象调用 Draw 方法。 虽然列表中的每个对象都具有声明类型 Shape,但调用的将是运行时类型(该方法在每个派生类中的重写版本)。

C#

public class Shape{    // A few example members     public int X { get; private set; }    public int Y { get; private set; }    public int Height { get; set; }    public int Width { get; set; }    // Virtual method     public virtual void Draw()    {        Console.WriteLine("Performing base class drawing tasks");    }}class Circle : Shape{    public override void Draw()    {        // Code to draw a circle...        Console.WriteLine("Drawing a circle");        base.Draw();    }}class Rectangle : Shape{    public override void Draw()    {        // Code to draw a rectangle...        Console.WriteLine("Drawing a rectangle");        base.Draw();    }}class Triangle : Shape{    public override void Draw()    {        // Code to draw a triangle...        Console.WriteLine("Drawing a triangle");        base.Draw();    }}class Program{    static void Main(string[] args)    {        // Polymorphism at work #1: a Rectangle, Triangle and Circle         // can all be used whereever a Shape is expected. No cast is         // required because an implicit conversion exists from a derived          // class to its base class.        System.Collections.Generic.List shapes = new System.Collections.Generic.List();        shapes.Add(new Rectangle());        shapes.Add(new Triangle());        shapes.Add(new Circle());        // Polymorphism at work #2: the virtual method Draw is         // invoked on each of the derived classes, not the base class.         foreach (Shape s in shapes)        {            s.Draw();        }        // Keep the console open in debug mode.        Console.WriteLine("Press any key to exit.");        Console.ReadKey();    }}/* Output:    Drawing a rectangle    Performing base class drawing tasks    Drawing a triangle    Performing base class drawing tasks    Drawing a circle    Performing base class drawing tasks */

在 C# 中,每个类型都是多态的,因为包括用户定义类型在内的所有类型都继承自 Object。

关注对象原则:调用子类还是父类的方法,取决于创建的对象是子类对象还是父类对象

多态性概述

虚成员

 派生类的设计器可以选择是否

重写基类中的虚拟成员。

继承最接近的基类方法而不重写它

定义隐藏基类实现的成员的新非虚实现

 派生成员必须使用 override 关键字显式指示该方法将参与虚调用。 以下代码提供了一个示例:

C#

public class BaseClass{    public virtual void DoWork() { }    public virtual int WorkProperty    {        get { return 0; }    }}public class DerivedClass : BaseClass{    public override void DoWork() { }    public override int WorkProperty    {        get { return 0; }    }}

 当派生类重写某个虚拟成员时,即使该派生类的实例被当作基类的实例访问,也会调用该成员。 以下代码提供了一个示例:

C#

DerivedClass B = new DerivedClass();B.DoWork();  // Calls the new method.BaseClass A = (BaseClass)B;A.DoWork();  // Also calls the new method.

 有关详细信息,请参阅使用 Override 和 New 关键字进行版本控制(C# 编程指南)。 接口提供另一种方式来定义将实现留给派生类的方法或方法集。 有关详细信息,请参阅接口(C# 编程指南)。

使用新成员隐藏基类成员

 new 关键字放置在要替换的类成员的返回类型之前。 以下代码提供了一个示例:

C#

public class BaseClass{    public void DoWork() { WorkField++; }    public int WorkField;    public int WorkProperty    {        get { return 0; }    }}public class DerivedClass : BaseClass{    public new void DoWork() { WorkField++; }    public new int WorkField;    public new int WorkProperty    {        get { return 0; }    }}

 例如:

C#

DerivedClass B = new DerivedClass();B.DoWork();  // Calls the new method.BaseClass A = (BaseClass)B;A.DoWork();  // Calls the old method.

阻止派生类重写虚拟成员

 如果类 A 声明了一个虚拟成员,类 B 从 A 派生,类 C 从类 B 派生,则类 C 继承该虚拟成员,并且可以选择重写它,而不管类 B 是否为该成员声明了重写。 以下代码提供了一个示例:

C#

public class A{    public virtual void DoWork() { }}public class B : A{    public override void DoWork() { }}

 这需要在类成员声明中的 override 关键字前面放置 sealed 关键字。 以下代码提供了一个示例:

C#

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

 即使它们转换为类型 B 或类型 A,它对于 C 的实例仍然是虚拟的。 通过使用 new 关键字,密封的方法可以由派生类替换,如下面的示例所示:

C#

public class D : C{    public new void DoWork() { }}

 如果使用类型为 C、B 或 A 的变量访问 D 的实例,对 DoWork 的调用将遵循虚拟继承的规则,即把这些调用传送到类 C 的 DoWork 实现。

从派生类访问基类虚拟成员

 以下代码提供了一个示例:

C#

public class Base{    public virtual void DoWork() {/*...*/ }}public class Derived : Base{    public override void DoWork()    {        //Perform Derived's work here         //...         // Call DoWork on base class         base.DoWork();    }}

有关详细信息,请参阅 base。

@@##@@ 说明

 允许基类行为发生使得派生类能够集中精力实现特定于派生类的行为。 未调用基类实现时,由派生类负责使它们的行为与基类的行为兼容。

 以上就是C#  多态性的内容,更多相关内容请关注PHP中文网(www.php.cn)!

说明

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432291.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 06:26:50
下一篇 2025年12月17日 06:26:59

相关推荐

  • C# 字符串中多个连续空格转为一个空格

    #region 字符串中多个连续空格转为一个空格 /// /// 字符串中多个连续空格转为一个空格 /// /// 待处理的字符串 /// 合并空格后的字符串 public static string MergeSpace(string str) { if (str != string.Empty …

    好文分享 2025年12月17日
    000
  • C# 自定义 implicit和explicit转换

    explicit 和 implicit 属于转换运算符,如用这两者可以让我们自定义的类型支持相互交换explicti 表示显式转换,如从 a -> b 必须进行强制类型转换(b = (b)a)implicit 表示隐式转换,如从 b -> a 只需直接赋值(a = b) 隐式转换可以让我…

    2025年12月17日
    000
  • C# 多线程参数传递

    1、通过实体类来传递(可以传递多个参数与获取返回值),demo如下: 需要在线程中调用的函数: namespace ThreadParameterDemo{ public class FunctionClass { public static string TestFunction(string n…

    2025年12月17日
    000
  • C# 下载带进度条代码(普通进度条)

    /// /// 下载带进度条代码(普通进度条) /// /// 网址 /// 下载后文件名为 /// 报告进度的处理(第一个参数:总大小,第二个参数:当前进度) /// True/False是否下载成功 public static bool DownLoadFile(string URL, stri…

    2025年12月17日 好文分享
    000
  • C# 可以利用反射给只读属性赋值吗?

    结论:可以 验证demo如下: using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.Drawing;using System.Linq;usi…

    2025年12月17日 好文分享
    000
  • C# 接口作用的深入理解

    原文地址:http://www.php.cn/ 假设我们公司有两种程序员:VB程序员,指的是用VB写程序的程序员,用clsVBProgramer这个类表示;Delphi程序员指的是用Delphi写程序的程序员,用clsDelphiProgramer这个类来表示。每个类都有一个WriteCode()方…

    好文分享 2025年12月17日
    000
  • C# 对象比较(值类型、引用类型)

    #region 引用对象比较 /// /// 引用对象比较 /// /// /// /// public static bool CompareObject(object objA, object objB) { bool flag = false; if (objA == null || objB…

    好文分享 2025年12月17日
    000
  • C# datatable中重复数据去重

    #region datatable去重 /// /// datatable去重 /// /// 需要去重的datatable /// 依据哪些列去重 /// public static DataTable GetDistinctTable(DataTable dtSource, params str…

    好文分享 2025年12月17日
    000
  • C# WCF DataContractSerializer 类

         原文地址:点击打开链接       DataContractSerializer 类      使用提供的数据协定,将类型实例序列化和反序列化为 XML 流或文档。 无法继承此类。      命名空间:  System.Runtime.Serialization        程序集:  …

    好文分享 2025年12月17日
    000
  • C# 实体类序列化与反序列化二 (DataContractSerializer)

    using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Runtime.Serialization;using System.IO;using System.Xml;…

    好文分享 2025年12月17日
    000
  • C# 将一个对象转换为指定类型

    原文地址:点击打开链接 适用:普通的对象,并且有默认的无参数构造函数 #region 将一个对象转换为指定类型 /// /// 将一个对象转换为指定类型 /// /// 待转换的对象 /// 目标类型 /// 转换后的对象 public static object ConvertToObject(o…

    好文分享 2025年12月17日
    000
  • C# Json 序列化与反序列化二

    /// /// 将对象转换为 JSON 字符串 /// /// /// /// public static string ScriptSerialize(T input) { string _jsonString = string.Empty; if (input != null) { JavaSc…

    好文分享 2025年12月17日
    000
  • C# 根据KeyEventArgs与组合键字符串相互转换

    /// 快捷键相关的类 /// public static class HotKeyInfo { /// /// 根据KeyEventArgs生成组合键字符串 /// /// /// public static string GetStringByKey(KeyEventArgs e) { if (…

    好文分享 2025年12月17日
    000
  • C# 内存管理

    windows使用一个系统:虚拟寻址系统,该系统把程序可用的内存地址映射到硬件内存中的实际地址上,这些任务完全由windows在后台管理。其实际结果是”位处理器上的每个进程都可以使用4gb的内存ˉ—无论计算机上实际有多少硬盘空间(在64位处理器上,这个数字会更大。这个4gb的内存实际上包含了程序的所…

    2025年12月17日
    000
  • C# DataTable中返回列中的最大值

    此处以表dt2中的keyIndex列(int类型)为例 1、通过linq来实现 int maxKeyIndex = dt2.AsEnumerable().Select(t => t.Field(“keyIndex”)).Max();linq语法:点击打开链接2、通过compute方法来实现 i…

    好文分享 2025年12月17日
    000
  • 使用 Override 和 New 关键字进行版本控制(C# 编程指南)

    原文地址:点击打开链接  这具有多方面的意义。例如,这意味着在基类中引入与派生类中的某个成员具有相同名称的新成员在 C# 中是完全支持的,不会导致意外行为。 它还意味着类必须显式声明某方法是要重写一个继承方法,还是一个隐藏具有类似名称的继承方法的新方法。 在 C# 中,派生类可以包含与基类方法同名的…

    好文分享 2025年12月17日
    000
  • C#基础知识整理 基础知识(19) 值类型的装箱和拆箱(二)

    如果代码中会造成编译器的反复装箱,可改为手动装箱,这样来使代码执行更快,看下面代码: //手动装箱 Int32 v = 5; //由于string.Format的参数是object类型,所以这里会造成三次装箱。 Console.WriteLine(string.Format(“{0},{1},{2}…

    好文分享 2025年12月17日
    000
  • C#基础知识整理 基础知识(20) 委托(一)

    委托和事件在c#中使用很多,尤其在进行窗体编程时,很多的操作都要通过委托和事件来处理和传递。这里详细解释下委托和事件的用法和原因,使我们在写代码时能更加明白一些。委托的使用事件是一种机制,事件是通过委托实现的,所以看明白了委托,对于事件也就能更好的理解了。先看下面一个小例子。这是描述不同语言的程序员…

    2025年12月17日
    000
  • C#基础知识整理 基础知识(21) 委托(二)

    看了前面一篇委托的用法中http://www.php.cn/,我们看到, public void Description(string programerName, DescriptionDelegate description) { description(programerName); } 这个…

    2025年12月17日
    000
  • C# 使用绑定句柄来减少进程的内存耗用

    许多应用程序中,绑定了一组类型(type)或者类型成员(从memberinfo派生),并将这些对象保存在某种形式的一个集合中。以后,会搜索这个集合,查找特定的对象,然后调用这个对象。这是一个很好的机制,但是有个小问题:type和memberinfo派生的对象需要大量的内存。如果一个应用程序容纳了太多…

    好文分享 2025年12月17日
    000

发表回复

登录后才能评论
关注微信