C#面向对象(OOPs)中的多态性

2024-03-09 03:36

本文主要是介绍C#面向对象(OOPs)中的多态性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文由 简悦 SimpRead 转码, 原文地址 mp.weixin.qq.com

C#面向对象(OOPs)中的多态性

概述:在编程语言和类型理论中,多态性是为不同类型的实体提供单个接口,或者使用单个符号来表示多个不同的类型。多态对象是能够呈现多种形式的对象。

编译时多态性(静态绑定或早期绑定或临时多态性):

编译时多态性存在两种类型,一种是运算符重载,另一种是方法重载

** 方法重载: **

允许一个类具有多个具有相同名称但参数不同的方法。这里只有参数的区别,不返回方法的类型。

using System;  public class Calculator  
{  public int Add(int a, int b)  {  return a + b;  }  // Overloaded method with different parameter types  public double Add(double a, double b)  {  return a + b;  }  // this is not possible can't have same paramter but different return type  public int Add(double a, double b)  {  return (int) a + b;  }  
}  class Program  
{  static void Main(string[] args)  {  Calculator calc = new Calculator();  int result1 = calc.Add(10, 20);  double result2 = calc.Add(3.5, 4.5);  Console.WriteLine("Result 1: " + result1);  Console.WriteLine("Result 2: " + result2);  }  
}
** 运算符重载:**

允许重新定义运算符以使用用户定义的类型。就像您可以在字符串类型中添加数字一样。

using System;  public class Complex  
{  public int Real { get; set; }  public int Imaginary { get; set; }  public Complex(int real = 0, int imaginary = 0)  {  Real = real;  Imaginary = imaginary;  }  // Overloading the '+' operator for adding two Complex objects  public static Complex operator +(Complex c1, Complex c2)  {  return new Complex(c1.Real + c2.Real, c1.Imaginary + c2.Imaginary);  }// Overloading the '!=' operator for adding two Complex objects  public static bool operator !=(Complex first, Complex second)  {  return !(first.Real == second.Real && first.Imaginary == second.Imaginary);  }  // Overloading the '==' operator for adding two Complex objects  public static bool operator ==(Complex first, Complex second)  {  return first.Real == second.Real && first.Imaginary == second.Imaginary;  }public override bool Equals(object? obj){if(obj == null || GetType() != obj.GetType())return false;Complex other = (Complex)obj;return Real == other.Real && Imaginary == other.Imaginary;}public override int GetHashCode(){return HashCode.Combine(Real, Imaginary);}public void Display()  {  Console.WriteLine($"Real: {Real}, Imaginary: {Imaginary}");  }  
}  class Program  
{  static void Main(string[] args)  {  Complex c1 = new Complex(3, 4);  Complex c2 = new Complex(5, 6);  Complex result = c1 + c2; // Using the overloaded '+' operator  result.Display();  }  
}

在上面的例子中,复杂类重载 + 和 != 运算符,使用运算符重载,您可以提供自定义的自然运算符重载 -、+、* 或 = 等。你在编译时定义的所有东西,这就是为什么这被称为编译时重载。默认情况下,在 c# 中,整数和字符串之间存在默认的 + 运算符重载。

运行时多态性(动态绑定或后期绑定):

** 方法覆盖:** :

当子类提供已在其超类或父类中定义的方法的特定实现时发生。子类或子类方法重写具有相同签名和相同返回类型的超类方法。

using System;  // Base class  
class Shape  
{  public virtual void Draw()  {  Console.WriteLine("Drawing a shape");  }  
}  // Derived class  
class Circle : Shape  
{  public override void Draw()  {  Console.WriteLine("Drawing a circle");  }  
}  class Program  
{  static void Main(string[] args)  {  // Runtime binding example  Shape shape = new Circle(); // Creating instance of derived class but of base class reference  shape.Draw(); // This will call the overridden method in Circle class at runtime  }  
}

方法覆盖:

  • 在此示例中,该类定义了一个虚拟方法 ShapeDraw() 。

  • 该类继承自该方法,并使用其自己的实现重写该方法 CircleShapeDraw() 。

  • 当子类提供已在其超类 中定义的方法的特定实现时,将发生方法重写。

  • Shape中的方法使用相同的签名和返回类型重写 in 中的方法。Draw()CircleDraw()Shape

** 运行时绑定:**
  • 在该方法中,创建一个实例 ,并将其分配给 类型的引用。MainCircleShape

  • 这是多态性 **(子类型多态性)的一个例子,后面的 i 节将对此进行解释 **,其中派生类 () 对象可以被视为其基类 () 类型。CircleShape

  • 在运行时,当调用时,C# 运行时系统将确定对象 () 的实际类型,并执行 中重写的方法。shape.Draw()CircleCircle

  • 这称为运行时绑定或后期绑定,其中在运行时根据对象的实际类型决定调用哪个方法。

** 参数多态性(泛型编程):**

** 泛 型:** 允许编写函数或类,这些函数或类可以在编译时对各种数据类型进行操作,而无需知道特定类型。它通过对类型进行抽象来提供灵活性和代码可重用性。

using System;  public class Stack<T>  
{  private T[] elements;  private int top;  public Stack(int size)  {  elements = new T[size];  top = -1;  }  public void Push(T item)  {  if (top == elements.Length - 1)  {  Console.WriteLine("Stack overflow!");  return;  }  elements[++top] = item;  }  public T Pop()  {  if (top == -1)  {  Console.WriteLine("Stack underflow!");  return default(T);  }  return elements[top--];  }  public void Print()  {  Console.WriteLine("Stack elements:");  for (int i = top; i >= 0; i--)  {  Console.WriteLine(elements[i]);  }  }  
}  class Program  
{  static void Main(string[] args)  {  Stack<int> intStack = new Stack<int>(5);  intStack.Push(1);  intStack.Push(2);  intStack.Push(3);  intStack.Print();  Console.WriteLine("Popped item: " + intStack.Pop());  Stack<string> stringStack = new Stack<string>(3);  stringStack.Push("Hello");  stringStack.Push("World");  stringStack.Print();  Console.WriteLine("Popped item: " + stringStack.Pop());  }  
}

在此示例中:

  • 我们定义了一个使用泛型类型参数的类Stack

  • 该类可以使用创建类实例时指定的任何数据类型。

  • 我们演示了如何使用 int 和 string 类型来使用类。Stack, Stack

  • 通过使用泛型,该类允许相同的实现使用不同的数据类型,从而提供灵活性和代码可重用性。

** 亚型多态性(Inclusion Polymorphism):**

方法覆盖: 使派生类的对象能够被视为其基类的对象,从而允许多态行为。当派生类继承自基类并提供其自己的某些方法的实现时,就会发生这种情况,从而重写基类方法的行为。

using System;  // Interface  
public interface IShape  
{  void Draw();  
}  // Class implementing the interface  
public class Circle : IShape  
{  public void Draw()  {  Console.WriteLine("Drawing a circle");  }  
}  // Another class implementing the interface  
public class Rectangle : IShape  
{  public void Draw()  {  Console.WriteLine("Drawing a rectangle");  }  
}  class Program  
{  static void Main(string[] args)  {  IShape shape1 = new Circle();  IShape shape2 = new Rectangle();  shape1.Draw(); // Calls the Draw method of Circle  shape2.Draw(); // Calls the Draw method of Rectangle  }  
}
  • 类Circle和类RectangleI都实现接口,并提供自己的ShapeDraw方法实现。

  • 在运行时,我们创建 Circle 和 Rectangle 类的对象,并将它们分配给 IShape 类型的变量。

大多数设计模式使用子类型多态来实现灵活和鲁棒的类设计,这是一种非常强大的多态行为,如策略设计模式或观察者设计模式或装饰器设计模式等。

** 强制多态性(铸造):**

强制多态性,也称为隐式转换,是指编译器或运行时环境将一种数据类型自动转换为另一种数据类型。这种转换是在安全且合乎逻辑的情况下进行的,从而允许更灵活和方便的编程。

using System;  class Program  
{  static void Main(string[] args)  {  int num1 = 10;  double num2 = num1; // Implicit conversion from int to double  Console.WriteLine("num1: " + num1); // Outputs: num1: 10  Console.WriteLine("num2: " + num2); // Outputs: num2: 10.0  }  
}

在此示例中:

  • 我们有一个初始化的整数变量num1,值为 10。

  • 我们将 num1的值赋值给一个双精度变量num2。此赋值将触发从 int到 double的隐式转换。

  • 编译器会自动将int值转换为double值,因为这样做是安全的,不会损失精度。

  • 这种隐式转换允许我们将num2视为一个double,即使它最初是一个int,** 表明强制多态性。**

总结

面向对象编程中的多态性包括编译时运行时行为。编译时多态性涉及方法和运算符重载,允许基于参数和运算符的多种行为。

通过方法重写实现的运行时多态性使子类能够提供自己的超类方法实现。其他形式包括具有泛型的参数多态性、通过方法覆盖和接口的子类型多态性,以及用于隐式类型转换的强制多态性。这些概念有助于提高软件开发中的代码灵活性、可重用性和适应性。

这篇关于C#面向对象(OOPs)中的多态性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

2. c#从不同cs的文件调用函数

1.文件目录如下: 2. Program.cs文件的主函数如下 using System;using System.Collections.Generic;using System.Linq;using System.Threading.Tasks;using System.Windows.Forms;namespace datasAnalysis{internal static

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M

C# dateTimePicker 显示年月日,时分秒

dateTimePicker默认只显示日期,如果需要显示年月日,时分秒,只需要以下两步: 1.dateTimePicker1.Format = DateTimePickerFormat.Time 2.dateTimePicker1.CustomFormat = yyyy-MM-dd HH:mm:ss Tips:  a. dateTimePicker1.ShowUpDown = t

C#关闭指定时间段的Excel进程的方法

private DateTime beforeTime;            //Excel启动之前时间          private DateTime afterTime;               //Excel启动之后时间          //举例          beforeTime = DateTime.Now;          Excel.Applicat

C# 防止按钮botton重复“点击”的方法

在使用C#的按钮控件的时候,经常我们想如果出现了多次点击的时候只让其在执行的时候只响应一次。这个时候很多人可能会想到使用Enable=false, 但是实际情况是还是会被多次触发,因为C#采用的是消息队列机制,这个时候我们只需要在Enable = true 之前加一句 Application.DoEvents();就能达到防止重复点击的问题。 private void btnGenerateSh

C# double[] 和Matlab数组MWArray[]转换

C# double[] 转换成MWArray[], 直接赋值就行             MWNumericArray[] ma = new MWNumericArray[4];             double[] dT = new double[] { 0 };             double[] dT1 = new double[] { 0,2 };

C# Hash算法之MD5、SHA

MD5我们用的还是比较多的,一般用来加密存储密码。但是现在很多人觉MD5可能不太安全了,所以都用上了SHA256等来做加密(虽然我觉得都差不多,MD5还是能玩)。 还是跟上一篇说的一样,当一个算法的复杂度提高的同时肯定会带来效率的降低,所以SHA和MD5比较起来的话,SHA更安全,MD5更高效。 由于HASH算法的不可逆性,所以我认为MD5和SHA主要还是应用在字符串的"加密"上。 由于

ffmpeg面向对象-待定

1.常用对象 rtsp拉流第一步都是avformat_open_input,其入参可以看下怎么用: AVFormatContext *fmt_ctx = NULL;result = avformat_open_input(&fmt_ctx, input_filename, NULL, NULL); 其中fmt_ctx 如何分配内存的?如下 int avformat_open_input(

C#线程系列(1):BeginInvoke和EndInvoke方法

一、线程概述 在操作系统中一个进程至少要包含一个线程,然后,在某些时候需要在同一个进程中同时执行多项任务,或是为了提供程序的性能,将要执行的任务分解成多个子任务执行。这就需要在同一个进程中开启多个线程。我们使用 C# 编写一个应用程序(控制台或桌面程序都可以),然后运行这个程序,并打开 windows 任务管理器,这时我们就会看到这个应用程序中所含有的线程数,如下图所示。