C# virtual 关键字

2024-06-03 02:44

本文主要是介绍C# virtual 关键字,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • virtual 使用
  • Override 关键字
  • New 关键字
  • 何时使用 Override / New 关键字?
  • 不要在构造函数里调用虚函数

virtual 使用

c#的方法,默认为非虚方法,如果一个方法被声明为 virtual (虚方法),则继承该方法的任何类都可以实现它自己的版本。

public class BaseEngineer
{public virtual void Work(){Console.WriteLine("BaseEngineer.Work");}
}

virtual 方法可以通过 Overridenew 关键字来进行版本控制、重写.

Override 关键字

Override 用来重写基类的虚方法。如下例子中,override 关键字可以确保派生类 JuniorEngineer / SeniorEngineer 的任何对象将使用 work 的派生类版本,同时又可以通过 base 关键字访问基类的版本。

public class JuniorEngineer : BaseEngineer
{public override void Work(){base.Work();Console.WriteLine("=>JuniorEngineer.Work");}
}
public class SeniorEngineer : BaseEngineer
{public override void Work(){base.Work();Console.WriteLine("=>SeniorEngineer.Work");}
}var juniorEngineer = new JuniorEngineer();
juniorEngineer.Work();var seniorEngineer = new SeniorEngineer();
seniorEngineer.Work();
----------------------Output---------------------------
BaseEngineer.Work
=>JuniorEngineer.Work
BaseEngineer.Work
=>SeniorEngineer.Work

New 关键字

派生类的方法前面带有 new 关键字,则该方法被定义为独立于基类中的方法,用 new 关键字可以隐藏基类中的虚方法。

假如派生类中的方法前面没有 new / override 关键字,那么编译期会发出 Warning,并将该方法作为有 new 关键字去执行。

public class SuperEngineer : BaseEngineer
{public new void Work(){base.Work();Console.WriteLine("=>Eat a bug!");Console.WriteLine("=>Create lots of bugs!");}
}var superEngineer = new SuperEngineer();
superEngineer.Work();
----------------------Output---------------------------
BaseEngineer.Work
=>Eat a bug!
=>Create lots of bugs!

何时使用 Override / New 关键字?

从上面可以看到,无论使用 Override / New 中哪一个,派生类的实例似乎都可以重新定义方法内容,也可以调用base的方法,那么为何要有2个关键字呢?

上面的例子,我们稍微改动一点,所有 override / new 的实现中,都不再调用 base 的虚方法,并通过基类访问派生类实例:

public class JuniorEngineer : BaseEngineer
{public override void Work(){Console.WriteLine("=>JuniorEngineer.Work");}
}
public class SeniorEngineer : BaseEngineer
{public override void Work(){Console.WriteLine("=>SeniorEngineer.Work");}
}
public class SuperEngineer : BaseEngineer
{public new void Work(){Console.WriteLine("=>Eat a bug!");Console.WriteLine("=>Create lots of bugs!");}
}var enginners = new List<BaseEngineer>
{juniorEngineer,seniorEngineer,superEngineer
};foreach (var enginner in enginners)
{enginner.Work();
}--------------Output----------------------
=>JuniorEngineer.Work
=>SeniorEngineer.Work
BaseEngineer.Work

Junior / Senior 都如期工作,调用了派生类中的方法,而最后的 SuperEngineer 并没有执行派生类的方法,而是执行了基类的虚方法。

因为数组的类型为 BaseEngine,且派生类 SuperEngineer 使用了 new 关键字重新定义方法,因此最终执行的是基类虚方法。

不要在构造函数里调用虚函数

这一条是引自《Effective c#》,在构建对象的过程中调用虚方法会使程序表现出奇怪的行为,因为这个时候对象并没有完全构造好。将书中的例子改的更为详尽一些:

public class VirtualB
{protected VirtualB(){Console.WriteLine("VritualB constructor start");VFunc();Console.WriteLine("VritualB constructor end");}protected virtual void VFunc(){Console.WriteLine("VirtualB.VFunc");}
}
public class Derived : VirtualB
{private readonly string msg = "Set by initializer";public Derived(string msg){Console.WriteLine("Derived constructor start");this.msg = msg;VFunc();Console.WriteLine("Derived constructor end");}protected override void VFunc(){Console.WriteLine($"Derived: {msg}");}
}var d = new Derived("Constructed in main");----------------Output---------------------------
VritualB constructor start
Derived: Set by initializer
VritualB constructor end
Derived constructor start
Derived: Constructed in main
Derived constructor end

基类的构造函数调用了一个定义在本类中的虚函数,于是派生类实例在运行时调用的就是派生类的版本,即 Derived 的派生版本。

C# 在进入构造函数体之前,已经把该对象的所有成员变量初始化好了,即开发者声明每一个成员变量时写的所有初始化语句都得到了执行。

整个流程如下:

  1. msg 首先通过初始化语句赋值为 “Set by initializer”
  2. 开始执行构造函数,先执行 base 构造函数
    • base 构造函数调用派生类中的 VFunc 中的方法输出 “Derived: Set by initializer”
  3. 执行派生类 Derived 构造函数
    • msg 通过构造函数入参赋值为 “Constructed in main”
    • 再次调用 VFunc

可以看到,构造函数中调用虚方法,可能会导致程序结果不是我们所期望的,除非你清晰理解c#语言规范,否则这样会令程序可读性降低,甚至引起数据混乱。

这篇关于C# virtual 关键字的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1025811

相关文章

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

C#中Guid类使用小结

《C#中Guid类使用小结》本文主要介绍了C#中Guid类用于生成和操作128位的唯一标识符,用于数据库主键及分布式系统,支持通过NewGuid、Parse等方法生成,感兴趣的可以了解一下... 目录前言一、什么是 Guid二、生成 Guid1. 使用 Guid.NewGuid() 方法2. 从字符串创建

C# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式

《C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式》Markdown凭借简洁的语法、优良的可读性,以及对版本控制系统的高度兼容性,逐渐成为最受欢迎的文档格式... 目录为什么要将文档转换为 Markdown 格式使用工具将 Word 文档转换为 Markdown(.

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

Java 关键字transient与注解@Transient的区别用途解析

《Java关键字transient与注解@Transient的区别用途解析》在Java中,transient是一个关键字,用于声明一个字段不会被序列化,这篇文章给大家介绍了Java关键字transi... 在Java中,transient 是一个关键字,用于声明一个字段不会被序列化。当一个对象被序列化时,被