学习c#的第十三天

目录

C# 多态性

静态多态性

函数重载

运算符重载

动态多态性

virtual 和 abstract

抽象方法和虚方法的区别

重载(overload)和重写(override)

隐藏方法


C# 多态性

多态是同一个行为具有多个不同表现形式或形态的能力。

多态性意味着有多重形式。在面向对象编程范式中,多态性往往表现为"一个接口,多个功能"。

多态性可以是静态的或动态的。在静态多态性中,函数的响应是在编译时发生的。在动态多态性中,函数的响应是在运行时发生的。

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

多态就是同一个接口,使用不同的实例而执行不同操作,如图所示:

现实中,比如我们按下 F1 键这个动作:

  • 如果当前在 Flash 界面下弹出的就是 AS 3 的帮助文档;
  • 如果当前在 Word 下弹出的就是 Word 帮助;
  • 在 Windows 下弹出的就是 Windows 帮助和支持。

同一个事件发生在不同的对象上会产生不同的结果。

静态多态性

在编译时,函数和对象的连接机制被称为早期绑定,也被称为静态绑定。C# 提供了两种技术来实现静态多态性。分别为:

函数重载

在C#中,函数重载指的是在同一个类中可以定义多个具有相同名称但参数列表不同的函数。通过函数重载,可以根据不同的参数列表来调用合适的函数。

下面是一个简单的示例,演示了如何在C#中使用函数重载:

using System;

public class DateUtils
{
    // 函数重载
    public string FormatDate(DateTime dt)
    {
        return dt.ToString("yyyy/MM/dd HH:mm:ss.fff");
    }

    public string FormatDate(DateTime dt, string format)
    {
        return dt.ToString(format);
    }
}

public class Program
{
    public static void Main()
    {
        DateUtils utils = new DateUtils();

        DateTime now = DateTime.Now;

        // 使用不同的重载函数进行日期格式化
        string formattedDate1 = utils.FormatDate(now); // 使用第一个重载函数
        string formattedDate2 = utils.FormatDate(now, "yyyy-MM-dd HH:mm:ss"); // 使用第二个重载函数

        // 输出格式化后的日期
        Console.WriteLine("格式化日期1: " + formattedDate1);
        Console.WriteLine("格式化日期2: " + formattedDate2);
    }
}

在上面的示例中,DateUtils 类中定义了两个名为 FormatDate 的函数。第一个函数接受一个 DateTime 类型的参数,并将日期格式化为特定的格式。第二个函数接受一个 DateTime 类型的参数和一个表示日期格式的字符串参数,并根据传入的格式对日期进行格式化。这两个函数拥有相同的名称但参数列表不同。

运算符重载

C#允许我们重定义或重载内置运算符,以便在用户自定义类型上使用它们。运算符重载通过关键字operator后面跟着运算符的符号来定义。

下面是一些常见的运算符重载的示例:

1、算术运算符:

  • +:重载为public static T operator +(T operand1, T operand2)
  • -:重载为public static T operator -(T operand1, T operand2)
  • *:重载为public static T operator *(T operand1, T operand2)
  • /:重载为public static T operator /(T operand1, T operand2)
  • %:重载为public static T operator %(T operand1, T operand2)

2、比较运算符:

  • ==:重载为public static bool operator ==(T operand1, T operand2)
  • !=:重载为public static bool operator !=(T operand1, T operand2)
  • <:重载为public static bool operator <(T operand1, T operand2)
  • >:重载为public static bool operator >(T operand1, T operand2)
  • <=:重载为public static bool operator <=(T operand1, T operand2)
  • >=:重载为public static bool operator >=(T operand1, T operand2)

3、逻辑运算符:

  • !:重载为public static bool operator !(T operand)
  • &:重载为public static bool operator &(T operand1, T operand2)
  • |:重载为public static bool operator |(T operand1, T operand2)

4、位运算符:

  • ~:重载为public static T operator ~(T operand)
  • &:重载为public static T operator &(T operand1, T operand2)
  • |:重载为public static T operator |(T operand1, T operand2)
  • ^:重载为public static T operator ^(T operand1, T operand2)

5、赋值运算符:

  • =:重载为public static void operator =(T destination, T source)
  • +=:重载为public static T operator +=(T operand1, T operand2)
  • -=:重载为public static T operator -=(T operand1, T operand2)
  • *=:重载为public static T operator *=(T operand1, T operand2)
  • /=:重载为public static T operator /=(T operand1, T operand2)
  • %=:重载为public static T operator %=(T operand1, T operand2)

这些示例只是一些常见的运算符重载示例,你可以根据自己的需求重载其他运算符。需要注意的是,并非所有运算符都可以被重载,具体可重载的运算符列表请参考C#官方文档。

代码示例

using System;

public class Complex
{
    public double Real { get; set; }
    public double Imaginary { get; set; }

    public Complex(double real, double imaginary)
    {
        Real = real;
        Imaginary = imaginary;
    }

    // 重载加法运算符 "+"
    public static Complex operator +(Complex c1, Complex c2)
    {
        return new Complex(c1.Real + c2.Real, c1.Imaginary + c2.Imaginary);
    }

    // 重载减法运算符 "-"
    public static Complex operator -(Complex c1, Complex c2)
    {
        return new Complex(c1.Real - c2.Real, c1.Imaginary - c2.Imaginary);
    }

    // 重载乘法运算符 "*"
    public static Complex operator *(Complex c1, Complex c2)
    {
        double real = c1.Real * c2.Real - c1.Imaginary * c2.Imaginary;
        double imaginary = c1.Real * c2.Imaginary + c1.Imaginary * c2.Real;
        return new Complex(real, imaginary);
    }

    // 重载复数的输出格式
    public override string ToString()
    {
        return $"{Real} + {Imaginary}i";
    }
}

class Program
{
    static void Main()
    {
        Complex c1 = new Complex(1, 2);
        Complex c2 = new Complex(3, 4);

        Complex sum = c1 + c2; // 使用重载的加法运算符
        Console.WriteLine($"总和: {sum}");

        Complex difference = c1 - c2; // 使用重载的减法运算符
        Console.WriteLine($"差别: {difference}");

        Complex product = c1 * c2; // 使用重载的乘法运算符
        Console.WriteLine($"相乘: {product}");
    }
}

当上面的代码被编译和执行时,它会产生下列结果:

总和: 4 + 6i
差别: -2 + -2i
相乘: -5 + 10i

动态多态性

在C#中,使用抽象类和虚方法可以实现动态多态性。

C# 允许您使用关键字 abstract 创建抽象类,用于提供接口的部分类的实现。当一个派生类继承自该抽象类时,实现即完成。抽象类包含抽象方法,抽象方法可被派生类实现。派生类具有更专业的功能。

请注意,下面是有关抽象类的一些规则:

  • 抽象类不能被实例化,只能被用作其他类的基类。
  • 抽象类中可以包含抽象方法,这些方法没有具体的实现,而是由派生类来实现。
  • 通过在类定义前面放置关键字 sealed,可以将类声明为密封类。当一个类被声明为 sealed 时,它不能被继承。抽象类不能被声明为 sealed。

下面的程序演示了一个抽象类:

using System;

// 定义抽象类
public abstract class Shape
{
    public string Name { get; set; }

    // 定义抽象方法
    public abstract double CalculateArea();
}

// 继承自抽象类的具体类:圆形
public class Circle : Shape
{
    public double Radius { get; set; }

    // 实现抽象方法
    public override double CalculateArea()
    {
        return Math.PI * Radius * Radius;
    }
}

// 继承自抽象类的具体类:正方形
public class Square : Shape
{
    public double SideLength { get; set; }

    // 实现抽象方法
    public override double CalculateArea()
    {
        return SideLength * SideLength;
    }
}

class Program
{
    static void Main()
    {
        Circle circle = new Circle { Name = "圆形", Radius = 3 };
        Console.WriteLine("圆形的面积: " + circle.CalculateArea());  // 输出结果:"圆形的面积: 28.274333882308138"

        Square square = new Square { Name = "正方形", SideLength = 4 };
        Console.WriteLine("正方形的面积: " + square.CalculateArea());  // 输出结果:"正方形的面积: 16"
    }
}

 另外,虚方法是在基类中,可以使用关键字virtual来声明一个方法为虚方法,表示它可以被派生类重写。而在派生类中,可以使用关键字override来重写基类中的虚方法。

通过使用抽象类和虚方法,可以实现代码的可扩展性和灵活性。抽象类提供了一种统一的接口,定义了需要实现的方法,而具体的实现则由派生类来完成。虚方法允许在派生类中重写方法,实现不同的行为。

以下是一个简单的代码示例,演示了如何使用抽象类和虚方法来实现动态多态性。

using System;

// 定义一个抽象类
public abstract class Shape
{
    // 定义一个虚方法
    public virtual double CalculateArea()
    {
        return 0;
    }
}

// 派生类1:矩形
public class Rectangle : Shape
{
    public double Width { get; set; }
    public double Height { get; set; }

    // 实现抽象方法
    public override double CalculateArea()
    {
        return Width * Height;
    }
}

// 派生类2:圆形
public class Circle : Shape
{
    public double Radius { get; set; }

    // 实现抽象方法
    public override double CalculateArea()
    {
        return Math.PI * Radius * Radius;
    }
}

class Program
{
    static void Main()
    {
        // 创建一个矩形对象
        Shape rect = new Rectangle { Width = 5, Height = 10 };
        Console.WriteLine("矩形的面积:" + rect.CalculateArea());

        // 创建一个圆形对象
        Shape circle = new Circle { Radius = 3 };
        Console.WriteLine("圆形的面积:" + circle.CalculateArea());
    }
}

当上面的代码被编译和执行时,它会产生下列结果:

矩形的面积:50
圆形的面积:28.274333882308138

virtual 和 abstract

  • virtual关键字用于修饰方法,表示该方法在基类中有一个默认实现,但派生类可以选择性地重写它。通常情况下,virtual方法会提供一个默认实现,但可以在派生类中进行修改或扩展。即使派生类没有重写virtual方法,它仍然会使用基类中的默认实现。
  • abstract关键字用于修饰方法或类,表示它们没有具体的实现。抽象方法只有定义,没有实现,派生类必须重写它们,并且提供具体的实现。抽象类本身也不能被实例化,只能作为基类被其他类继承。

需要注意的是,抽象方法只能存在于抽象类中,而不是普通的类。如果一个类包含抽象方法,则该类本身必须被声明为抽象类。而虚方法可以存在于任何类中,无论是抽象类还是普通类。

总之,virtual和abstract关键字都用于实现多态性,在派生类中重新定义父类的方法。virtual方法允许派生类选择重写并提供新的实现,而abstract方法则要求派生类必须提供具体的实现。另外,抽象类本身不能被实例化,只能作为其他类的基类使用。

抽象方法和虚方法的区别

  1. 抽象方法(abstract method)必须在抽象类中声明,并且没有具体的实现部分。派生类必须重写并提供具体的实现,否则派生类将无法被实例化。
  2. 虚方法(virtual method)在基类中有一个默认的实现,但也可以在派生类中进行重写。派生类可以选择性地覆盖虚方法,或者直接使用基类的默认实现。
  3. 抽象方法和虚方法都可以供派生类重写以定制化行为,但抽象方法必须在派生类中进行重写,而虚方法可以选择性地进行重写。

重载(overload)和重写(override)

1、重载(overload): 在同一个作用域(一般指一个类)的两个或多个方法函数名相同,参数列表不同的方法叫做重载,它们有三个特点(俗称两必须一可以):

  •  方法名必须相同
  •  参数列表必须不相同
  •  返回值类型可以不相同

2、重写(override):子类中为满足自己的需要来重复定义某个方法的不同实现,需要用 override 关键字,被重写的方法必须是虚方法,用的是 virtual 关键字。它的特点是(三个相同):

  • 相同的方法名
  • 相同的参数列表
  • 相同的返回值

代码示例

using System;

public class Calculator
{
    // 方法重载,参数列表不同
    public int Add(int num1, int num2)
    {
        return num1 + num2;
    }

    public double Add(double num1, double num2)
    {
        return num1 + num2;
    }
}

public class Shape
{
    // 虚方法
    public virtual void Draw()
    {
        Console.WriteLine("正在绘制形状。。。");
    }
}

public class Circle : Shape
{
    // 重写基类的虚方法
    public override void Draw()
    {
        Console.WriteLine("正在绘制圆。。。");
    }
}

class Program
{
    static void Main()
    {
        Calculator calculator = new Calculator();
        int result1 = calculator.Add(3, 4);              // 调用第一个Add方法
        double result2 = calculator.Add(2.5, 3.7);       // 调用第二个Add方法

        Console.WriteLine("int加法的结果: " + result1); // 输出结果:"int加法的结果: 7"
        Console.WriteLine("double加法的结果: " + result2); // 输出结果:"double加法的结果: 6.2"

        Shape shape1 = new Shape();
        shape1.Draw();         // 输出结果:"正在绘制形状。。。"

        Shape shape2 = new Circle();
        shape2.Draw();         // 输出结果:"正在绘制圆。。。"
    }
}

隐藏方法

当在派生类中定义一个与基类中同名的方法时,会发生方法隐藏。使用关键字 new 可以实现方法隐藏。

using System;

public class Shape
{
    public void Draw()
    {
        Console.WriteLine("正在绘制形状。。。");
    }
}

public class Circle : Shape
{
    public new void Draw()
    {
        Console.WriteLine("正在绘制圆。。。");
    }
}

class Program
{
    static void Main()
    {
        Shape shape1 = new Shape();
        shape1.Draw();         // 输出结果:"正在绘制形状。。。"

        Shape shape2 = new Circle();
        shape2.Draw();         // 输出结果:"正在绘制形状。。。"(调用的是基类的方法)

        Circle circle = new Circle();
        circle.Draw();         // 输出结果:"正在绘制圆。。。"
    }
}

在上述代码中,Shape 类定义了一个 Draw() 方法,Circle 类继承自 Shape 并定义了一个同名的 Draw() 方法并使用 new 关键字进行方法隐藏。

在 Main() 方法中,首先创建了一个 Shape 对象 shape1,调用其 Draw() 方法,输出结果为 "正在绘制形状。。。"。接着创建了一个 Circle 对象 shape2,将其赋值给 Shape 类型的变量 shape2,再次调用 shape2.Draw() 方法,输出结果仍然为 ""正在绘制形状。。。"",这是因为变量的静态类型是 Shape,所以调用的是基类的方法。最后,创建了一个 Circle 对象 circle,直接调用其 Draw() 方法,输出结果为 "正在绘制圆。。。",这是因为直接通过派生类的实例调用方法时,会调用派生类中隐藏的方法。

需要注意的是,方法隐藏并不是方法重写。如果想要实现方法的多态性,应该使用方法重写(override)而不是隐藏(new)。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2023年山东省安全员A证证考试题库及山东省安全员A证试题解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2023年山东省安全员A证证考试题库及山东省安全员A证试题解析是安全生产模拟考试一点通结合&#xff08;安监局&#xff09;特种作业人员操作证考试大纲和&#xff08;质检局&#xff09;特种设备作业人员上岗证考试大…

一文讲透IGBT工作原理 | 百能云芯

在实际应用中最流行和最常见的电子元器件是双极结型晶体管 BJT 和 MOS管。 IGBT实物图电路符号图 你可以把 IGBT 看作 BJT 和 MOS 管的融合体&#xff0c;IGBT具有 BJT 的输入特性和 MOS 管的输出特性。与 BJT 或 MOS管相比&#xff0c;绝缘栅双极型晶体管 IGBT 的优势在于它提…

项目文章:oxBS揭示复发性膀胱癌的DNA甲基化和羟甲基化变化并鉴定预测PD-L1表达标记物

近日&#xff0c;徐州市中心医院&#xff08;徐州医科大学徐州临床学院&#xff09;史振铎等为第一作者、韩从辉教授为通讯作者在《Biomarker Research》杂志发表题为“Integrative multi-Omics analysis depicts the methylome and hydroxymethylome of recurrent bladder can…

印染污水处理设备的物理法分类有哪些设备?

印染污水处理设备的物理法分类主要有以下几种设备&#xff1a; 筛滤截留法&#xff1a;主要采用筛网、格栅、滤池与微滤机等设备&#xff0c;用于去除污水中的悬浮物和漂浮物。重力分离法&#xff1a;主要采用沉砂池、沉淀池、隔油池与气浮机等设备&#xff0c;利用重力或浮力…

使用量子玻尔兹曼机推进机器学习:新范式

一、说明 量子玻尔兹曼机&#xff08;QBM&#xff09;是量子物理学和机器学习的前沿融合。通过利用叠加和纠缠等量子特性的力量&#xff0c;QBM 可以同时探索多个解决方案&#xff0c;使其异常擅长解决复杂问题。它使用量子位&#xff08;量子计算的构建模块&#xff09;以传统…

表白墙完善(数据库,前端,后端Servlet),再谈Cookie和Session。以及一个关于Cookie的练习小程序

目录 表白墙引入数据库 再谈Cookie和session 得到Cookie ​编辑 设置Cooie 使用Cookie编写一个登入的小界面 表白墙引入数据库 1.先引入数据库的依赖&#xff08;驱动包&#xff09;&#xff0c;5.1.49 pom.xml中&#xff0c;在之前的两个之前&#xff0c;再去添加一个 &…

RT-DETR算法优化改进:轻量级上采样CARAFE算子 | 注意力机制大作战

💡💡💡本文独家改进: 上采样操作CARAFE,具有感受野大、内容感知、轻量级、计算速度快等优点,引入RT-DETR二次创新; 1)代替Upsample进行使用; 推荐指数:五星 RT-DETR魔术师专栏介绍: https://blog.csdn.net/m0_63774211/category_12497375.html ✨✨✨魔改创…

Seatunnel及web搭建流程

准备工作 Java环境 要求java8或者java11&#xff0c;并设置JAVA_HOME&#xff0c;如果JAVA_HOME无效&#xff0c;需要设置为有效状态。 echo ${JAVA_HOME} 创建安装软件目录 sudo mkdir -p /opt/seatunnel/backend sudo mkdir -p /opt/seatunnel/web 下载软件 从https://…

新手小白看过来——带你快速入门跨境电商

近几年来&#xff0c;国内外贸交易是越来越火爆&#xff0c;其中跨境电商成为了2023年的热门风口行业&#xff0c;尽管现在做跨境电商的从业者有很多&#xff0c;但仍然有许多0基础小白想通过跨境电商获取人生的第一桶金&#xff0c;那么新手应该如何在跨境电商领域取得成功呢&…

jedis连接redis

package com.wsd;import redis.clients.jedis.Jedis;import java.io.IOException; import java.io.InputStream; import java.util.HashMap; import java.util.Map; import java.util.Properties;public class Redis {public static void main(String[] args) {//读取properti…

侧面多级菜单(一个大类、一个小类、小类下多个物体)

效果&#xff1a; 说明&#xff1a; 左右侧面板使用Animator组件控制滑入滑出。左侧面板中&#xff0c;左的左里面是大类&#xff0c;左的右有绿色的小类&#xff0c;绿色的小类下有多个真正的UI图片按钮。 要点&#xff1a; 结合了一点EasyGridBuilderPro插件的UI元素&…

天津市专业大数据培训班,大数据就业岗位的多样性

大数据技术应用广泛&#xff0c;几乎涉及到了各个行业和领域。毕业后&#xff0c;我们可以选择从事大数据工程师、数据分析师、数据科学家等职业&#xff0c;也可以选择进入金融、医疗、电商等行业进行数据分析和决策支持。 大数据就业岗位多样 大数据培训所涉及的就业岗位有…

【从入门到起飞】JavaSE—IO工具包(Commons-io,Hutool) (2)

&#x1f38a;专栏【JavaSE】 &#x1f354;喜欢的诗句&#xff1a;天行健&#xff0c;君子以自强不息。 &#x1f386;音乐分享【如愿】 &#x1f384;欢迎并且感谢大家指出小吉的问题&#x1f970; 文章目录 &#x1f33a;工具包Commons-io⭐使用步骤&#x1f6f8;新建一个文…

国内也可以做伦敦金的吗?给你答案

虽然伦敦金是国际市场上的贵金属投资方式&#xff0c;但国内投资者同样有机会参与其中&#xff0c;最简单的方式就是通过香港的持牌平台&#xff0c;在网上开设一个账号&#xff0c;并往其中注入一定的资金之后&#xff0c;就可以展开交易。 国内投资者可以炒伦敦金&#xff0c…

Spring Bean的生命周期详细梳理

1. 理解Bean的生命周期 1.1 生命周期的各个阶段 在Spring IOC容器中&#xff0c;Bean的生命周期大致如下&#xff1a; 实例化&#xff1a;当启动Spring应用时&#xff0c;IOC容器就会为在配置文件中声明的每个<bean>创建一个实例。属性赋值&#xff1a;实例化后&#…

Python大语言模型实战-利用MetaGPT框架自动开发一个游戏软件(附完整教程)

实现功能 MetaGPT是一个应用在软件开发领域的多智能体框架&#xff0c;其主要创新点在于将SOP标准流水线和Agent结合在了一起&#xff0c;使得拥有不同技能的Role之间配合完成一项较为复杂的任务。本文将用一个案例来演示整个流程。 实现代码 项目地址&#xff1a;https://gi…

MapApp 地图应用

1. 简述 1.1 重点 1&#xff09;更好地理解 MVVM 架构 2&#xff09;更轻松地使用 SwiftUI 框架、对齐、动画和转换 1.2 资源下载地址: Swiftful-Thinking:https://www.swiftful-thinking.com/downloads 1.3 项目结构图: 1.4 图片、颜色资源文件图: 1.5 启动图片配置图: 2. Mo…

(Matalb时序预测)WOA-BP鲸鱼算法优化BP神经网络的多维时序回归预测

目录 一、程序及算法内容介绍&#xff1a; 基本内容&#xff1a; 亮点与优势&#xff1a; 二、实际运行效果&#xff1a; 三、部分代码&#xff1a; 四、完整代码数据说明手册&#xff1a; 一、程序及算法内容介绍&#xff1a; 基本内容&#xff1a; 本代码基于Matalb平台…

微服务测试怎么做

开发团队越来越多地选择微服务架构而不是单体结构&#xff0c;以提高应用程序的敏捷性、可扩展性和可维护性。随着决定切换到模块化软件架构——其中每个服务都是一个独立的单元&#xff0c;具有自己的逻辑和数据库&#xff0c;通过 API 与其他单元通信——需要新的测试策略和新…

python 基础语法学习 (二)

多变量赋值 当你在Python中进行多变量赋值时&#xff0c;你可以在一行代码中同时为多个变量分配值。这种方法可以简化代码并提高可读性。下面是一些关于Python多变量赋值的基本知识&#xff1a; 基本赋值&#xff1a;你可以使用等号&#xff08;&#xff09;将一个值分配给一…