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

相关文章

C#提取PDF表单数据的实现流程

《C#提取PDF表单数据的实现流程》PDF表单是一种常见的数据收集工具,广泛应用于调查问卷、业务合同等场景,凭借出色的跨平台兼容性和标准化特点,PDF表单在各行各业中得到了广泛应用,本文将探讨如何使用... 目录引言使用工具C# 提取多个PDF表单域的数据C# 提取特定PDF表单域的数据引言PDF表单是一

C#实现添加/替换/提取或删除Excel中的图片

《C#实现添加/替换/提取或删除Excel中的图片》在Excel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更加美观,下面我们来看看如何在C#中实现添加/替换/提取或删除E... 在Excandroidel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

c# checked和unchecked关键字的使用

《c#checked和unchecked关键字的使用》C#中的checked关键字用于启用整数运算的溢出检查,可以捕获并抛出System.OverflowException异常,而unchecked... 目录在 C# 中,checked 关键字用于启用整数运算的溢出检查。默认情况下,C# 的整数运算不会自

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

C# 读写ini文件操作实现

《C#读写ini文件操作实现》本文主要介绍了C#读写ini文件操作实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录一、INI文件结构二、读取INI文件中的数据在C#应用程序中,常将INI文件作为配置文件,用于存储应用程序的

C#实现获取电脑中的端口号和硬件信息

《C#实现获取电脑中的端口号和硬件信息》这篇文章主要为大家详细介绍了C#实现获取电脑中的端口号和硬件信息的相关方法,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 我们经常在使用一个串口软件的时候,发现软件中的端口号并不是普通的COM1,而是带有硬件信息的。那么如果我们使用C#编写软件时候,如