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#语言规范,否则这样会令程序可读性降低,甚至引起数据混乱。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/21205.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android Media Framework(一)OpenMAX 框架简介

学习开源代码最快的方式是先阅读它的文档&#xff0c;再查看它的头文件&#xff0c;最后研读代码实现并进行编译调试。Android早期引入OpenMAX IL作为使用音视频编解码器的标准接口&#xff0c;了解Android Media框架的底层运行原理要从OMX IL开始。在这一节&#xff0c;我们将…

qt c++ 随机数 获取mac地址

目录 获取mac地址 c 随机数 c pro设置&#xff1a; QT core gui network 获取mac地址 #include <QCoreApplication> #include <QNetworkInterface> #include <QDebug>int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);// 获取所有网络…

本机安装深度学习库cuda11.8,cudnn8.6和tensorRT8.5

https://blog.csdn.net/qq_46107892/article/details/131453019 首先是安装cuda11.8 wget https://developer.download.nvidia.com/compute/cuda/repos/ubuntu2004/x86_64/cuda-ubuntu2004.pinsudo mv cuda-ubuntu2004.pin /etc/apt/preferences.d/cuda-repository-pin-600wg…

44-3 waf绕过 - WAF绕过方法

环境准备: 43-5 waf绕过 - 安全狗简介及安装-CSDN博客然后在安装pikachu靶场:构建完善的安全渗透测试环境:推荐工具、资源和下载链接_渗透测试靶机下载-CSDN博客一、首先验证云WAF是否存在于靶场(老师的靶场是部署在阿里云) 靶场地址:http://127.0.0.1/pikachu-master/v…

游戏找不到steam_api64.dll如何解决,全面解析原因及解决方法

在现代游戏中&#xff0c;Steam平台已经成为了玩家们下载、安装和玩游戏的主要渠道之一。然而&#xff0c;有些玩家可能会遇到一个问题&#xff0c;即游戏找不到steam_api64.dll文件。这个问题可能会导致游戏无法正常运行或启动。本文将详细介绍如何解决这个问题&#xff0c;帮…

23、linux系统文件和日志分析

linux文件系统与日志分析 文件时存储在硬盘上的&#xff0c;硬盘上的最小存储单位是扇区&#xff0c;每个扇区大大小是512字节。 inode&#xff1a;元信息&#xff08;文件的属性 权限&#xff0c;创建者&#xff0c;创建日期等&#xff09; block&#xff1a;块&#xff0c…

ZDH-数据管理模块

目录 主题 项目源码 预览地址 安装包下载地址 数据管理服务 数据资源管理 数据资源权限 数据资源血缘 总结 感谢支持 主题 本篇文章主要介绍ZDH-数据管理服务及应用场景 项目源码 zdh_web: GitHub - zhaoyachao/zdh_web: 大数据采集,抽取平台 预览地址 后台管理…

Mac安装pytorch(二)

书接上回&#xff0c;配置好了pytorch环境后&#xff0c;看看是否真的能用 终端输入一下代码&#xff1a; import torch xtorch.rand(3,4) print(x) 出现这些后表明安装完成&#xff0c;可使用 接下来在pycharm中使用 打开设置

JavaScript数组应用

检测数据类型 1.typeof()可以检测基本数据类型&#xff0c;但是在检测null时会返回object。另外它不能检测负责的数据类型&#xff0c;如正则表达式对象 2.constructor可以检测绝大部分数据的类型&#xff0c;但是不能检测null和underfined的数据类型 3.toString()方法&#x…

视频监控平台AS1000:通过网络SDK接入松下视频监控设备(Panasonic监控摄像机) 的源代码的函数和功能介绍及分享

目录 一、视频监控平台介绍 1、概述 2、视频接入能力介绍 3、功能介绍 二、PANASONIC网络摄像机 1、产品种类与定位 2、规格参数 3、功能特点 4、环境适应性 5、网络功能 6、其他特性 三、代码和解释 1、代码和注释 2、函数功能说明 &#xff08;1&#xff09;处…

python学习:语法(1)

目录 内置函数&#xff1a; 输出函数&#xff1a; 转义字符 二进制与字符编码 python中的保留字和标识符 保留字 标识符 变量的定义和使用 变量由三部分组成 变量的多次赋值 运算符 数学运算符&#xff1a; 加减乘除 取余运算符 幂运算符 &#xff1a; 赋值运算符…

TypeScript 中的命名空间

1. 命名空间的概念 命名空间是 TypeScript 提供的一种组织代码的方式&#xff0c;它类似于其他编程语言中的模块化系统&#xff0c;但有一些不同之处。命名空间可以包含变量、函数、类等&#xff0c;并且可以嵌套使用&#xff0c;从而更好地组织和管理代码。 2. 定义命名空间…

网络网络层

data: 2024/5/25 14:02:20 周六 limou3434 叠甲&#xff1a;以下文章主要是依靠我的实际编码学习中总结出来的经验之谈&#xff0c;求逻辑自洽&#xff0c;不能百分百保证正确&#xff0c;有错误、未定义、不合适的内容请尽情指出&#xff01; 文章目录 1.协议结构2.封装分离3.…

2023年全国职业院校技能大赛(高职组)“云计算应用”赛项赛卷7(私有云)

#需要资源&#xff08;软件包及镜像&#xff09;或有问题的&#xff0c;可私聊博主&#xff01;&#xff01;&#xff01; #需要资源&#xff08;软件包及镜像&#xff09;或有问题的&#xff0c;可私聊博主&#xff01;&#xff01;&#xff01; #需要资源&#xff08;软件包…

24年护网工具,今年想参加护网的同学要会用

24年护网工具集 吉祥学安全知识星球&#x1f517;http://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247483727&idx1&sndb05d8c1115a4539716eddd9fde4e5c9&chksmc0e47813f793f105017fb8551c9b996dc7782987e19efb166ab665f44ca6d900210e6c4c0281&scene21…

msgpack-c使用指南

msgpack MessagePack 是一种高效的二进制序列化格式&#xff0c;可在多种语言之间交换数据&#xff0c;但它更快、更小。小整数被编码为单个字节&#xff0c;而短字符串除了字符串本身之外只需要一个额外的字节。 msgpackcxx使用指南 msgpackcxx是一个纯头文件库。 安装指南…

513.找树左下角的值

给定一个二叉树&#xff0c;在树的最后一行找到最左边的值。 示例 1: 示例 2: 思路&#xff1a; 深度最大的叶子结点一定是最后一行。 优先左边搜索&#xff0c;记录深度最大的叶子节点&#xff0c;此时就是树的最后一行最左边的值 代码&#xff1a; class Solution:def fi…

【C#】自定义List排序规则的两种方式

目录 1.系统排序原理 2.方式一&#xff1a;调用接口并重写 3.方式二&#xff1a;传排序规则函数做参数 1.系统排序原理 当我们对一个List<int>类型的数组如list1排序时&#xff0c;一个轻松的list1.sort();帮我们解决了问题 但是在实际应用过程中&#xff0c;往往我们…

【Python】Python异步编程

Python 异步编程 异步编程 异步编程是一种编程范式&#xff0c;通过非阻塞的方式执行任务&#xff0c;允许程序在等待某些操作&#xff08;如I/O操作、网络请求、数据库查询等&#xff09;完成时&#xff0c;继续执行其他任务。这与同步编程&#xff08;或阻塞编程&#xff09…

如何使用python检查网络设备的状态

使用Python来定期检查网络设备的状态通常涉及几个步骤&#xff0c;包括网络连通性测试、发送和接收SNMP&#xff08;简单网络管理协议&#xff09;请求、SSH&#xff08;安全外壳协议&#xff09;连接或Telnet连接等。以下是一些建议的方法&#xff1a; 网络连通性测试&#xf…