news 2026/4/15 3:42:33

C#核心:继承

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C#核心:继承

继承的基本概念

一个类A继承另一个类B:

1、A将会继承类B的所有成员

2、A类将拥有B类的所有特征和行为

被继承的类称为:父类、基类、超类
继承的类称为:子类、派生类

注意:子类可以有自己的特征和行为

特点说明
1. 单根性C# 不支持多重继承,一个类只能直接继承一个父类。但可以通过接口实现多行为。
2. 传递性如果A继承BB继承C,那么A也间接继承C的成员。

基本语法

class 类名 : 被继承的类名 { }

继承类的案例说明

class Teacher { //名字 public string name; //工号 public int number; //介绍名字 public void SpeakName() { Console.WriteLine(name); } } //继承Teacher类 class TeachingTeacher : Teacher { //科目 public string subject; //介绍科目 public void SpeakSubject() { Console.WriteLine(subject+"老师"); } } class ChineseTeacher : TeachingTeacher { public void Skill() { Console.WriteLine("一行白鹭上青天"); } }

这里你可以看出ChineseTeacher都是继承了TeachingTeacher的属性和行为——然后直接通过new 一个类名创建对应实例

class Program { static void Main(string[] args) { // 创建基类实例 TeachingTeacher tt = new TeachingTeacher(); tt.name = "唐老狮"; tt.number = 1; tt.SpeakName(); tt.subject = "Unity"; tt.SpeakSubject(); // 创建派生类实例 ChineseTeacher ct = new ChineseTeacher(); ct.name = "唐老师"; ct.number = 2; ct.subject = "语文"; ct.SpeakName(); ct.SpeakSubject(); ct.Skill(); } }

继承类的访问权限修饰

关键字:protected

不希望外部访问,子类可以访问

class Teacher { …………//这里使用protected访问权限 protected int number; } //继承Teacher类 class TeachingTeacher : Teacher { number=10;//可以用 } class Program { static void Main(string[] args) { // 创建基类实例 TeachingTeacher tt = new TeachingTeacher(); tt.number = 1;//这里会报错,无法访问 } }

如果父类中的成员是private,那么子类和外部都无法访问该成员

里氏替换原则

基本概念

任何父类出现的地方,子类都可以代替:即父类容器转载子类对象(因为子类对象包含了父类对象全部内容)

作用:方便进行对象存储和管理

案例使用说明

class GameObject { } class Player : GameObject { public void PlayerAtk() { Console.WriteLine("玩家攻击"); } } class Moster : GameObject { public void MosterAtk() { Console.WriteLine("怪物攻击"); } } class Boss : GameObject { public void BossAtk() { Console.WriteLine("Boss攻击"); } }
class Program { static void Main(string[] args) { Console.WriteLine("里氏替换原则"); // 里氏替换原则:用父类容器装载子类对象 GameObject player = new Player(); GameObject monster = new Monster(); GameObject boss = new Boss(); GameObject[] objects = new GameObject[] { new Player(), new Monster(), new Boss() }; } }

但需要注意,此处子类对象都是GameObject类型,无法使用子类对象中特有的方法和行为。所以可能还需要强制转换相关参数。

is和as

当用GameObject类型 统一存储继承类 Player类,可以先用is判断该对象是否为Player类,再强制转换GameObject类为Player类

GameObject player = new Player(); //is判断一个对象是否是指定的对象 //返回值bool if (player is Player) { //as:将一个对象转换为指定类对象 //返回值:指定类型对象 //成功返回指定类对象 失败返回null Player p = player as Player; } //可以正常使用Player类的功能了 p.PlayerAtk();

使用数组管理基类

//实际使用时和数组配合使用多 方便进行对象存储和管理 GameObject[] objects = new GameObject[] { new Player(), new Moster(), new Boss() }; //遍历objects数组 来判断类和执行类 for (int i = 0; i < objects.Length; i++) { if (objects[i] is Player) { (player as Player).PlayerAtk(); } else if (objects[i] is Moster) { (player as Moster).MosterAtk(); } else if (objects[i] is Boss) { (player as Boss).BossAtk(); } }

继承中的构造函数

基本概念

当声明一个子类对象时,先执行父类的构造函数再执行子类构造函数。

子类实例化时 默认调用无参 父类没有无参构造就会报错

GameObject { public GameObject() { Console.WriteLine("GameObject的构造函数"); } } class Player : GameObject { public Player() { Console.WriteLine("Player的构造函数"); } } class MainPlayer : Player { public MainPlayer() { Console.WriteLine("Player的构造函数"); } }

这里构造子类函数MainPlayer

class Program { static void Main(string[] args) { Console.WriteLine("继承中的构造函数"); MainPlayer mp = new MainPlayer(); } }

打印结果:

父类的无参构造函数很重要

class Father { public Father (int a) { } } class Son : Father//报错 { }

就算Son也是有参构造函数,父类也会报错——因为无参构造函数很重要!!

base调用指定父类构造

class Father { //父类没有无参构造 public Father(int i) { Console.WriteLine("Father构造"); } } class Son : Father { public Son(int i) : base(i)//成功运行 { Console.WriteLine("Son的一个参数的构造"); } public Son(int i, string str) //报错!! { Console.WriteLine("Son的两个参数的构造"); } }

解决方法是:1、给父类添加上无参构造函数 2、给子类添加上重载构造函数

以下是方法2

class Father { public Father(int i) { Console.WriteLine("Father构造"); } } class Son : Father { public Son(int i) : base(i)//3、调用父类 { Console.WriteLine("Son的一个参数的构造"); } public Son(int i, string str) : this(i)//2、这里会调用Son(int i) : base(i),也是先进再执行 { Console.WriteLine("Son的两个参数的构造"); } } class Program { static void Main(string[] args) { Son s2 = new Son(20, "Hello"); // 1、调用 Son(int, string) } }
打印结果: Father构造 Son的一个参数的构造 Son的两个参数的构造

万物之父

关键词:object

是一个基类 可以装任何东西

强制转换方式

引用类型和里氏替换原则一样用is和as

值类型使用括号强转

String类型有专门的函数

// 引用类型 object o = new Son(); // 用 is as 来判断和转换即可 if (o is Son) { (o as Son).Speak(); } // 值类型 object o2 = 1f; // 用强转 float f1 = (float)o2; // 特殊的 string 类型 object str = "123123"; string str2 = str.ToString();

装箱拆箱

值类型和object之间发生的

装箱:把值类型用引用类型存储 栈内存移到堆内存中

拆箱:把引用类型存储的值类型取出来 堆内存移到栈内存中

int a =10; //装箱 object o = a; //拆箱 int b = (int)o;

密封类

关键字:sealed

作用是让类无法被继承结扎

保证程序的规范性,安全性

sealed class Father { } class son:Fathter//报错 { }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/9 13:05:21

基于DeepSeek-OCR-WEBUI的多语言OCR实践:支持表格、公式与手写体识别

基于DeepSeek-OCR-WEBUI的多语言OCR实践&#xff1a;支持表格、公式与手写体识别 1. 引言&#xff1a;复杂场景下的OCR新范式 随着企业数字化进程加速&#xff0c;文档自动化处理需求日益增长。传统OCR技术在面对多语言混排、复杂版面、手写体、数学公式和表格结构时&#xf…

作者头像 李华
网站建设 2026/4/9 17:50:42

HY-MT1.5-1.8B服务监控:Prometheus集成部署实战案例

HY-MT1.5-1.8B服务监控&#xff1a;Prometheus集成部署实战案例 1. 引言 随着大语言模型在翻译任务中的广泛应用&#xff0c;如何高效部署并实时监控模型服务的运行状态成为工程落地的关键环节。HY-MT1.5-1.8B作为一款轻量级高性能翻译模型&#xff0c;在边缘设备和实时场景中…

作者头像 李华
网站建设 2026/4/10 10:16:13

Youtu-2B异常检测:对话异常模式识别

Youtu-2B异常检测&#xff1a;对话异常模式识别 1. 引言 1.1 技术背景与问题提出 随着大语言模型&#xff08;LLM&#xff09;在智能客服、虚拟助手和自动化内容生成等场景中的广泛应用&#xff0c;确保对话系统的稳定性与安全性变得至关重要。Youtu-LLM-2B 作为腾讯优图实验…

作者头像 李华
网站建设 2026/4/4 2:40:42

Qwen2.5-0.5B低成本部署:CPU环境节省90%算力成本

Qwen2.5-0.5B低成本部署&#xff1a;CPU环境节省90%算力成本 1. 背景与技术选型动机 在当前大模型快速发展的背景下&#xff0c;越来越多开发者希望将AI能力集成到实际应用中。然而&#xff0c;主流大模型通常依赖高性能GPU进行推理&#xff0c;导致部署成本高、资源门槛高&a…

作者头像 李华
网站建设 2026/4/10 10:20:31

YOLO26显存不足怎么办?GPU显存优化实战解决方案

YOLO26显存不足怎么办&#xff1f;GPU显存优化实战解决方案 1. 问题背景与挑战 在使用最新 YOLO26 官方版训练与推理镜像 进行深度学习模型训练和推理时&#xff0c;许多开发者会遇到一个常见但棘手的问题&#xff1a;GPU 显存不足&#xff08;Out of Memory, OOM&#xff09…

作者头像 李华
网站建设 2026/4/12 22:45:49

MinerU文档理解精度实测:部署教程+真实样例结果展示

MinerU文档理解精度实测&#xff1a;部署教程真实样例结果展示 1. 引言 在当前信息爆炸的时代&#xff0c;智能文档理解技术正成为提升办公效率、加速科研阅读的关键工具。传统的OCR技术虽能提取文字&#xff0c;但在语义理解、图表解析和上下文关联方面存在明显短板。为此&a…

作者头像 李华