查看原文
其他

.NET 开发都应掌握C#泛型的知识点

DotNet 2024-04-12

前言

C#泛型编程是.NET framework 2.0的新特征,它提高了代码安全性和代码重用的问题。

由于泛型必须指定类型,正确的使用可以避免了装箱拆箱的问题;使用了泛型后,可以将算法或组件泛型化,基于类型参数支持任意数据类型,从而提高了代码的重用性。

C#泛型编程在C#是很重要的特性,建议.NET开发熟练掌握它的基本知识点,本文将介绍它的几个知识点,供大家参考。

一、泛型类型和方法

泛型类型和方法允许在定义时不指定具体的类型,而是在使用时确定类型。泛型类型和方法可以增加代码的灵活性和可重用性。这是C#泛型最基本的功能。

示例代码:

public T GetMax<T>(T a, T b) where T : IComparable<T>
{
     return a.CompareTo(b) > 0 ? a : b;
}
// 泛型类型方法示例
public class MyGenericList<T>
{
     private List<T> items = new List<T>();
     public void Add(T item){
             items.Add(item);
     }
}

二、泛型约束

泛型约束允许限制泛型参数的类型,以增加类型安全性和代码约束。

可以使用关键字where在泛型方法或泛型类型上应用约束。大家可以参考前面的文章深入理解C#泛型:new与where关键字全解析

示例代码

public void Process<T>(T item) where T : IComparable<T>
{
  // 使用IComparable<T>接口进行比较
}
// 以上代码泛型参数 T 受到了 IComparable<T> 泛型约束。
//这个约束指定了传入的类型 T 必须实现 IComparable<T> 接口,
//以便在方法内部进行比较操作

三、泛型接口和委托

泛型接口和委托允许在定义接口和委托时使用泛型类型参数,使其更通用和灵活。

示例代码

 // 泛型接口示例
public interface IRepository<T>
{
    void Add(T item);
}
// 泛型委托示例
public delegate void MyGenericDelegate<T>(T item);

四、泛型类的继承和接口实现

泛型类可以继承其他泛型类或非泛型类,也可以实现泛型或非泛型接口。可以在派生类或实现类中指定具体的类型参数。

示例代码

public class MyGenericList<T> : List<T>
{
     // 实现泛型类List<T>
}
public class MyRepository<T> : IRepository<T>
{
  public void Add(T item)
  {
      // 实现IRepository<T>接口
  }
}
//以上是常用的各种helper类示例,熟悉吧!

五、泛型集合类

C#还提供了许多内置的泛型集合类来提升开发效率,例如List、Dictionary<TKey, TValue>等。泛型集合类提供了类型安全和高效的数据存储和检索。

示例代码

List<string> names = new List<string>();
names.Add("张山");
names.Add("李四");

Dictionary<int, string> example= new Dictionary<int, string>();
example.Add(1, "222");
example.Add(2, "333");

六、泛型委托和事件

泛型委托和事件允许定义具有泛型参数的委托类型和事件处理器。可以在使用时指定具体的类型参数。多用于订阅事件。

示例代码

// 定义一个泛型委托 MyGenericDelegate<T>,表示可以接受类型为 T 的参数的委托
public delegate void MyGenericDelegate<T>(T item);

// 定义一个泛型类 EventPublisher<T>,用于发布泛型事件
public class EventPublisher<T>
{
    // 声明一个泛型事件 MyEvent,事件处理器类型为 MyGenericDelegate<T>
    public event MyGenericDelegate<T> MyEvent;

    // 触发事件的方法,将传入的参数 item 传递给事件处理器
    public void RaiseEvent(T item)
{
        // 使用空值条件运算符 ?. 检查事件是否有订阅者,如果有,则调用委托的 Invoke 方法
        MyEvent?.Invoke(item);
    }
}

//调用 
public static void Main()
{
     // 创建事件处理器方法
       void MyEventHandler(int item)
       {
            Console.WriteLine($"Event handled: {item}");
        }

        // 实例化 EventPublisher<int> 类并订阅事件
        EventPublisher<int> publisher = new EventPublisher<int>();
        publisher.MyEvent += MyEventHandler;

        // 调用 RaiseEvent 方法来触发事件
        int eventData = 42;
        publisher.RaiseEvent(eventData);
 }

七、泛型约束和接口协变/逆变

C# 4.0引入了接口的协变和逆变,使得泛型类型参数可以在接口中进行协变(out)或逆变(in)。这主要解决之前泛型接口调用子类父类类型不同而出现的问题。协变允许将派生类型转换为基类型,逆变允许将基类型转换为派生类型。大家可以看这篇文章C#泛型的逆变协变之个人理解

//协变
public interface IAnimal<out T>
{
     T Get();
}
 //逆变   
public interface ICage<in T>
{
      void Add(T animal);
}

八、泛型默认值

泛型类型的默认值可以使用default(T)获取,无需了解具体的类型。默认值对于泛型方法或类中可能为null的引用类型或可空值类型非常有用。

public void Process<T>(T item)
{
    T defaultValue = default(T);
    if (item.Equals(defaultValue))
    {
        // 处理默认值情况
    }
}

结语

本文讲述了C#泛型编程的知识点及关键概念和用法。它们可以提高代码的可重用性、类型安全性和性能等。

希望本文对你有所收获,你还知道C#泛型的哪些知识点?欢迎留言讨论和吐槽。

转自:翔星

链接:cnblogs.com/xbhp/p/17593155.html

- EOF -

推荐阅读  点击标题可跳转
C# 12 中的 Primary Constructor.NET  Socket调试工具之Ava.SocketTool.NET 开源免费功能最全的商城项目

看完本文有收获?请转发分享给更多人

推荐关注「DotNet」,提升.Net技能 

点赞和在看就是最大的支持❤️

继续滑动看下一个
向上滑动看下一个

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存