C# 多态 派生类 abstract virtual new
- 静态多态
- 函数重载
- 运算符重载
- 动态多态
- abstract 和 virtual的区别
- 定义与用途:
- 成员实现:
- 继承与重写:
- 与接口的区别:
- 使用抽象类的好处主要体现在以下几个方面:
- 代码重用:
- 设计灵活性:
- 接口定义:
- 类型安全:
- 易于扩展和维护:
- 符合面向对象的设计原则:
- 综上所述,
- 派生类中调用基类成员
- 通过base关键字调用:
- 直接调用:
- 通过构造函数:
- 在C#中,new关键字有多个用途
- 对象实例化:
- 隐藏基类成员:
- 约束:
- 运算符重载:
- 初始化数组:
- 随对象一起初始化字段:
- 例子
静态多态
在编译时,函数和对象的连接机制被称为早期绑定,也被称为静态绑定。C# 提供了两种技术来实现静态多态性。分别为:
函数重载
同名不同参
运算符重载
动态多态
C# 允许您使用关键字 abstract 创建抽象类,用于提供接口的部分类的实现。当一个派生类继承自该抽象类时,实现即完成。抽象类包含抽象方法,抽象方法可被派生类实现。派生类具有更专业的功能。
请注意,下面是有关抽象类的一些规则:
您不能创建一个抽象类的实例。
您不能在一个抽象类外部声明一个抽象方法。
通过在类定义前面放置关键字 sealed,可以将类声明为密封类。当一个类被声明为 sealed 时,它不能被继承。抽象类不能被声明为 sealed。
总之,
类前声明abstruct就可以写出一个抽象类
当有一个定义在类中的函数需要在继承类中实现时,可以使用虚方法。
虚方法是使用关键字 virtual 声明的。
虚方法可以在不同的继承类中有不同的实现。
对虚方法的调用是在运行时发生的。
动态多态性是通过 抽象类 和 虚方法 实现的。
在C#(以及其他一些面向对象的编程语言中),override是一个关键字,用于指示一个方法在派生类中覆盖了基类中的同名方法。这意味着,当你使用派生类的对象并调用这个方法时,将执行派生类中的版本,而不是基类中的版本。
要使用override关键字,必须满足以下条件:
基类中存在一个虚方法(virtual)或抽象方法(abstract):基类中必须有一个具有相同签名(即相同的名称、参数列表和返回类型)的虚方法或抽象方法。
派生类继承自基类:包含override方法的类必须是从包含被覆盖方法的类继承而来的。
方法签名匹配:派生类中的方法必须与基类中被覆盖的方法具有相同的签名。
访问修饰符不能比基类方法更严格:派生类中方法的访问修饰符(如public、protected或internal)不能比基类中被覆盖的方法更严格。例如,如果基类中的方法是public的,那么派生类中的覆盖方法也必须是public的。
using System;
namespace PolymorphismApplication
{class Shape {protected int width, height;public Shape( int a=0, int b=0){width = a;height = b;}public virtual int area(){Console.WriteLine("父类的面积:");return 0;}}class Rectangle: Shape{public Rectangle( int a=0, int b=0): base(a, b){}public override int area (){Console.WriteLine("Rectangle 类的面积:");return (width * height); }}class Triangle: Shape{public Triangle(int a = 0, int b = 0): base(a, b){}public override int area(){Console.WriteLine("Triangle 类的面积:");return (width * height / 2); }}class Caller{public void CallArea(Shape sh){int a;a = sh.area();Console.WriteLine("面积: {0}", a);}} class Tester{static void Main(string[] args){Caller c = new Caller();Rectangle r = new Rectangle(10, 7);Triangle t = new Triangle(10, 5);c.CallArea(r);c.CallArea(t);Console.ReadKey();}}
}
abstract 和 virtual的区别
在C#中,abstract和virtual是两个用于类和成员修饰的关键字,它们各自有着特定的用途和区别。
定义与用途:
abstract关键字用于定义抽象类和抽象成员(包括方法、属性、索引器、事件)。抽象类是一种特殊的类,不能被实例化,只能被其他类继承。抽象成员在抽象类中没有具体的实现,必须在派生类中提供实现。
virtual关键字用于定义虚方法、虚属性、虚索引器或虚事件。虚成员可以在派生类中被重写(override)。这意味着派生类可以提供该成员的不同实现。
成员实现:
abstract修饰的方法或属性不能有实现,它们只是声明了一个约定,具体的实现必须在派生类中完成。
virtual修饰的方法或属性必须有方法实现(哪怕只有一对大括号)。派生类可以选择是否重写虚成员。
继承与重写:
如果一个类包含抽象方法,那么这个类必须被声明为抽象类。抽象方法必须在任何非抽象派生类中被重写。
虚方法可以被任何继承它的类重写,但不是必须的。如果派生类没有重写虚方法,那么将使用基类中的实现。
与接口的区别:
抽象类是对对象的抽象,可以包含抽象方法和非抽象方法,是对一组具有共同特征的对象的抽象。它不能被实例化,但可以通过继承来创建具体的对象。
接口是一种行为规范,只包含抽象方法和事件的声明,不包含任何实现。任何实现接口的类都必须提供接口中所有方法的实现。
总的来说,abstract和virtual在C#中都是用于实现多态性的重要机制,但它们在定义、实现、继承与重写方面有着明显的区别。abstract更强调强制派生类提供实现,而virtual则提供了一种可选的重写机制。
使用抽象类的好处主要体现在以下几个方面:
代码重用:
抽象类允许我们定义一组通用的方法和属性,这些方法和属性可以被多个派生类共享。通过继承抽象类,派生类可以自动获得这些通用方法和属性的实现,从而避免了在每个派生类中重复编写相同的代码。
设计灵活性:
抽象类为设计提供了更大的灵活性。通过将某些方法或属性声明为抽象成员(即没有具体实现的方法或属性),我们可以强制派生类提供这些成员的具体实现。这有助于确保派生类遵循一定的接口或协议,从而实现更加一致和可预测的行为。
接口定义:
抽象类可以作为接口的一种实现方式。与纯接口相比,抽象类可以包含部分实现,使得某些通用的功能可以在抽象类中实现,而派生类只需关注特定于它们自己的实现。
类型安全:
通过抽象类,我们可以限制哪些类可以作为派生类。只有实现了抽象类中所有抽象成员的类才能被实例化。这有助于确保类型安全,防止不正确的类继承和使用。
易于扩展和维护:
当需要添加新的功能或修改现有功能时,我们只需在抽象类中进行相应的更改,所有继承该抽象类的派生类都将自动获得这些更改。这大大简化了代码的扩展和维护过程。
符合面向对象的设计原则:
使用抽象类有助于遵循面向对象的设计原则,如开闭原则(对扩展开放,对修改封闭)和里氏替换原则(子类必须能够替换其基类)。这些原则有助于构建更加健壮、可维护和可扩展的软件系统。
综上所述,
使用抽象类可以提高代码的重用性、设计灵活性、类型安全性以及易于扩展和维护性,同时也有助于遵循面向对象的设计原则。
派生类中调用基类成员
在派生类中调用基类成员的方法取决于成员的类型(字段、属性、方法)以及成员的访问修饰符。以下是几种常见的方式来在派生类中调用基类成员:
通过base关键字调用:
当派生类需要访问或调用被覆盖(override)的基类方法或属性时,可以使用base关键字来引用基类中的成员。
base关键字用于访问基类的构造函数、字段、属性、方法等。
public class BaseClass
{ public void BaseMethod() { Console.WriteLine("This is the base method."); }
} public class DerivedClass : BaseClass
{ public new void BaseMethod() // 注意这里使用了new关键字来隐藏基类方法 { base.BaseMethod(); // 调用基类中被覆盖的方法 Console.WriteLine("This is the derived method."); }
}
直接调用:
如果派生类没有覆盖基类的成员,那么可以直接调用这些成员,而无需使用base关键字。
public class DerivedClass : BaseClass
{ public void CallBaseMethod() { BaseMethod(); // 直接调用基类方法 }
}
通过构造函数:
在派生类的构造函数中,可以使用base关键字来调用基类的构造函数。
public class BaseClass
{ public BaseClass() { Console.WriteLine("BaseClass constructor called."); }
} public class DerivedClass : BaseClass
{ public DerivedClass() : base() // 调用基类的构造函数 { Console.WriteLine("DerivedClass constructor called."); }
}
请注意,如果基类成员是私有的(private),那么派生类无法直接访问它们。私有成员只能在定义它们的类内部被访问。如果基类成员是受保护的(protected),则派生类可以直接访问这些成员。如果基类成员是公共的(public)或受保护的,那么派生类可以像访问任何其他公共或受保护成员一样访问它们。
此外,当派生类使用new关键字隐藏基类中的成员时,虽然派生类成员和基类成员具有相同的名称,但它们实际上是两个不同的成员。在这种情况下,如果需要调用基类中被隐藏的成员,就必须使用base关键字。
在C#中,new关键字有多个用途
它们在不同上下文中具有不同的意义。以下是new关键字在C#中的一些主要用途:
对象实例化:
最常见的用法是创建对象实例。例如:
MyClass myObject = new MyClass();
这里,new MyClass()用于调用MyClass的构造函数并创建该类的一个新实例。
隐藏基类成员:
当在派生类中定义与基类成员同名的成员时,可以使用new关键字来显式地表示派生类成员隐藏了基类成员。这样做通常是为了避免基类成员的意外调用。
class Base
{ public void MyMethod() { /* ... */ }
} class Derived : Base
{ public new void MyMethod() { /* ... */ }
}
在这个例子中,Derived类中的MyMethod隐藏了Base类中的MyMethod。
约束:
在泛型类、接口或方法的定义中,new()约束用于指定类型参数必须具有无参数的公共构造函数。
public class MyClass<T> where T : new()
{ public T CreateInstance() { return new T(); }
}
这里,new()约束确保T类型有一个无参数的构造函数,因此可以在CreateInstance方法中创建它的实例。
运算符重载:
new运算符也可以被重载以用于自定义类型。这通常与资源管理和释放相关,尤其是在实现IDisposable接口时。但是,直接重载new运算符并不常见,因为C#提供了更具体的语法来管理资源(如using语句)。
初始化数组:
new用于创建数组实例,并可以初始化数组元素。
int[] myArray = new int[5] { 1, 2, 3, 4, 5 };
这里,new int[5]创建一个包含5个整数的数组,并通过花括号中的值进行初始化。
随对象一起初始化字段:
在对象实例化时,可以使用对象初始化器语法与new一起初始化对象的字段或属性。
var person = new Person
{ Name = "Alice", Age = 30
};
这里,new Person创建了一个Person对象,并使用对象初始化器语法设置了Name和Age属性。
请注意,new关键字的具体行为取决于它在哪里以及如何被使用。在使用new时,应确保你了解其当前上下文中的确切意义,以避免潜在的错误或混淆。
例子
using System;class TODO {static void Main() {Console.Write("Hello C#!\n");Tools tools = new Tools();int ans = tools.add(1, 2);Console.WriteLine(ans);Cars chaos = new Cars();Console.WriteLine("this thing is {0}.", chaos.SayName());chaos.x = 1;chaos.y = 2;chaos.z = 3;Console.WriteLine(chaos.Area());}
}class Tools {public int add(int x, int y) { return x + y; }public int add(int x, int y, int z) { return x + y + z; }
}abstract class Boxs {public int x;public int y;public int z;public int Area() {return x * y * x;}abstract public string SayName();virtual public void SayHello() { System.Console.Write("Hello!\n"); }private int total;
}class Cars: Boxs {public override string SayName() { return "CAR"; }public override void SayHello() {base.SayHello();}public new int Area() {return x + y + z;}
}
相关文章:
C# 多态 派生类 abstract virtual new
静态多态函数重载运算符重载 动态多态abstract 和 virtual的区别定义与用途:成员实现:继承与重写:与接口的区别: 使用抽象类的好处主要体现在以下几个方面:代码重用:设计灵活性:接口定义&#x…...
【爬虫基础】第10讲 urlerror的使用及捕获异常
URLError是Python中的一个异常类,用于处理与URL相关的错误。它是urllib.error模块中的一个类。 URLError通常在以下情况下被引发: 网络连接问题:例如无法连接到服务器、超时等。URL不正确:例如无效的URL、无法解析主机名等。服务…...
绍兴越城中墙建材蒸压加气混凝土砌块使用注意事项可送塔山府山北海蕺山城南稽山迪荡灵芝东湖皋埠马山斗门鉴湖东浦孙端陶堰富盛
绍兴越城中墙建材蒸压加气混凝土砌块使用注意事项可送塔山府山北海蕺山城南稽山迪荡灵芝东湖皋埠马山斗门鉴湖东浦孙端陶堰富盛 使用蒸压加气混凝土砌块时需要注意以下事项: 选择符合国家标准的产品:选购时应查看产品质量证明书,确保产品符合…...
吴渔夫:AI技术引领游戏产业革命,小团队有大作为
AI技术的突飞猛进,游戏产业正在经历一场前所未有的变革。中国网游先锋,火石控股创始人吴渔夫,近日在接受第一财经日报的采访,对AI在游戏制作中的应用和未来趋势有着深刻的见解。 吴渔夫指出,AI技术的引入极大地降低了游…...
深入探索C++对象模型(二)
类对象占用的空间 #include "pch.h" #include <iostream> using namespace std;class A {public: };//类对象所占用的空间 int main() {//std::cout << "Hello World!\n"; A obja;int ilen = sizeof(obja); cout << ilen << endl…...
【javaWeb 第三篇】Vue快速入门
VUE vue是一套前端框架,免除原生的js的DOM操作,简化书写 基于MVVM(model-view-viewmodel)思想,实现数据的双向绑定,将编程的关注放在数据上。 什么是框架: 框架相当于一个半成品,是一…...
非root用户安装git lfs(git大文件)命令记录
背景 最近在看LLAMA2的模型,想直接从Huggingface下载模型到本地,但是却发现服务器上没有安装git lfs命令。查询了一些资料完成了非root用户安装git lfs命令的操作,特此记录。 Git LFS下载与解压 下载 Git LFS 二进制文件 访问 Git LFS 发布…...
PTA 道路管制
乌拉乌拉国有n个城市和m条道路,城市编号为1∼n。由于乌拉乌拉国每一个城市都在创城(创建文明城市),因此,城市之间的道路通行施行道路交通管制: 已知从城市ui到城市vi的道路,需要时间ti。…...
自媒体用ChatGPT批量洗稿软件V5.9环境配置/软件设置教程【汇总】
大家好,我是淘小白~ 首先,感谢大家的支持~~ ChatGPT采集洗稿软件V5.9版本更新,此次版本更新修改增加了一些内容: 1、自定义多条指令,软件自动判断指令条数,进行输入 2、增加谷歌浏览多账号轮询…...
【WPF应用7】 基本控件-Grid 布局的详解与示例
引言 WPF(Windows Presentation Foundation)是.NET框架的一部分,它提供了一个用于创建桌面应用程序用户界面的框架。在WPF中,Grid布局是一个非常强大的布局工具,它允许开发者创建复杂的、响应迅速的用户界面布局。Grid…...
flink-connector-redis支持select查询
EN 1 项目介绍 基于bahir-flink二次开发,相对bahir调整的内容有: 1.使用Lettuce替换Jedis,同步读写改为异步读写,大幅度提升了性能 2.增加了Table/SQL API,增加select/维表join查询支持 3.增加关联查询缓存(支持增量与全量) 4…...
[密码学] 密码学基础
目录 一 为什么要加密? 二 常见的密码算法 三 密钥 四 密码学常识 五 密码信息威胁 六 凯撒密码 一 为什么要加密? 在互联网的通信中,数据是通过很多计算机或者通信设备相互转发,才能够到达目的地,所以在这个转发的过程中,如果通信包…...
上海:6月1日起取消企业复工复产白名单制
财经新闻5月29日消息:上海市人民政府关于印发《上海市加快经济恢复振兴行动计划》的通知。 《方案》包括千方百计缓解各类市场主体困难,全面有序推进复工复产和市场复工复产,多措并举稳外资稳外贸,大力促进消费加速复苏࿰…...
SpringBoot扩展篇:循环依赖源码链路
SpringBoot扩展篇:循环依赖源码链路 1. 相关文章2. 一个简单的Demo3. 流程图3.1 BeanDefinition的注册3.2 开始创建Bean3.3 从三级缓存获取Bean3.4 创建Bean3.5 实例化Bean3.6 添加三级缓存3.7 属性初始化3.8 B的创建过程3.9 最终流程 1. 相关文章 SpringBoot 源码…...
服务消费微服务
文章目录 1.示意图2.环境搭建1.创建会员消费微服务模块2.删除不必要的两个文件3.检查父子模块的pom.xml文件1.子模块2.父模块 4.pom.xml 添加依赖(刷新)5.application.yml 配置监听端口和服务名6.com/sun/springcloud/MemberConsumerApplication.java 创…...
uni-app纵向步骤条
分享一下项目中自封装的步骤条,存个档~ 1. 话不多说,先看效果 2. 话还不多说,上代码 <template><!-- 获取一个数组,结构为[{nodeName:"流程发起"isAudit:falsetime:"2024-02-04 14:27:35"otherDat…...
【JavaEE -- 文件操作IO有关面试题】
文件操作IO有关面试题 1.查找硬盘上的文件位置1.1 思路1.2 执行代码 2. 实现文件复制2.1 思路2.2 代码执行 3. 打印搜索的词的文件路径3.1 思路3.2 代码执行 1.查找硬盘上的文件位置 给定一个文件名,去指定的目录中进行搜索,找到文件名匹配的结果&#…...
Open WebUI大模型对话平台-适配Ollama
什么是Open WebUI Open WebUI是一种可扩展、功能丰富、用户友好的大模型对话平台,旨在完全离线运行。它支持各种LLM运行程序,包括与Ollama和Openai兼容的API。 功能 直观的界面:我们的聊天界面灵感来自ChatGPT,确保了用户友好的体验。响应…...
[2021]Zookeeper getAcl命令未授权访问漏洞概述与解决
今天在漏洞扫描的时候蹦出来一个zookeeper的漏洞问题,即使是非zookeeper的节点,或者是非集群内部节点,也可以通过nc扫描2181端口,获取极多的zk信息。关于漏洞的详细描述参考apache zookeeper官方概述:CVE-2018-8012: A…...
vscode添加gitee
1.创建仓库 2.Git 全局设置 3.初始化仓库 2.1 打开vscode打开需要上传到给git的代码文件 2.2.点击左边菜单第三个的源代码管理->初始化仓库 4.点击加号暂存所有更改 5.添加远程仓库 5.1 添加地址,回车 5.2 填写库名,回车 6.提交和推送 6.1 点击✔提交…...
RestClient
什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端,它允许HTTP与Elasticsearch 集群通信,而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级ÿ…...
【WiFi帧结构】
文章目录 帧结构MAC头部管理帧 帧结构 Wi-Fi的帧分为三部分组成:MAC头部frame bodyFCS,其中MAC是固定格式的,frame body是可变长度。 MAC头部有frame control,duration,address1,address2,addre…...
循环冗余码校验CRC码 算法步骤+详细实例计算
通信过程:(白话解释) 我们将原始待发送的消息称为 M M M,依据发送接收消息双方约定的生成多项式 G ( x ) G(x) G(x)(意思就是 G ( x ) G(x) G(x) 是已知的)࿰…...
【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器
——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的一体化测试平台,覆盖应用全生命周期测试需求,主要提供五大核心能力: 测试类型检测目标关键指标功能体验基…...
微信小程序 - 手机震动
一、界面 <button type"primary" bindtap"shortVibrate">短震动</button> <button type"primary" bindtap"longVibrate">长震动</button> 二、js逻辑代码 注:文档 https://developers.weixin.qq…...
多模态大语言模型arxiv论文略读(108)
CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...
推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材)
推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材) 这个项目能干嘛? 使用 gemini 2.0 的 api 和 google 其他的 api 来做衍生处理 简化和优化了文生图和图生图的行为(我的最主要) 并且有一些目标检测和切割(我用不到) 视频和 imagefx 因为没 a…...
【学习笔记】erase 删除顺序迭代器后迭代器失效的解决方案
目录 使用 erase 返回值继续迭代使用索引进行遍历 我们知道类似 vector 的顺序迭代器被删除后,迭代器会失效,因为顺序迭代器在内存中是连续存储的,元素删除后,后续元素会前移。 但一些场景中,我们又需要在执行删除操作…...
【从零开始学习JVM | 第四篇】类加载器和双亲委派机制(高频面试题)
前言: 双亲委派机制对于面试这块来说非常重要,在实际开发中也是经常遇见需要打破双亲委派的需求,今天我们一起来探索一下什么是双亲委派机制,在此之前我们先介绍一下类的加载器。 目录 编辑 前言: 类加载器 1. …...
数据结构:递归的种类(Types of Recursion)
目录 尾递归(Tail Recursion) 什么是 Loop(循环)? 复杂度分析 头递归(Head Recursion) 树形递归(Tree Recursion) 线性递归(Linear Recursion)…...
