Introduction
C# is a versatile and powerful programming language, and one of its key features is generics. Generics provide a way to design classes, interfaces, methods, and delegates with a placeholder for the data type, enabling code reusability and ensuring type safety. In this article, we will explore the concept of generics in C# with detailed examples, showcasing how they enhance code flexibility and maintainability.
Understanding Generics
Generics in C# allow you to create classes, interfaces, methods, or delegates with a placeholder for the data type that will be specified when the code is used. This enables you to write flexible and reusable code while ensuring type safety at compile-time.
Benefits of Generics
	- Code Reusability: Generics allow you to create components and algorithms that can work with different data types without sacrificing type safety or performance.
- Type Safety: Generics provide compile-time type checking, ensuring that the code is type-safe without the need for explicit casting or conversion at runtime.
- Performance: Generics improve performance by eliminating the need for boxing and unboxing operations, which occur when dealing with value types in non-generic collections.
Examples of Generics in C#
1. Generic Classes
public class GenericList<T>
{
    private List<T> _list = new List<T>();
    public void Add(T item)
    {
        _list.Add(item);
    }
    public T Get(int index)
    {
        return _list[index];
    }
}
// Usage
GenericList<string> stringList = new GenericList<string>();
stringList.Add("Hello, World!");
string output = stringList.Get(0);
In this example, GenericList<T> is a generic class that can work with any data type. You can create instances of GenericList for different types, ensuring type safety and code reusability.
2. Generic Methods
public T FindMax<T>(T[] array) where T : IComparable<T>
{
    if (array == null || array.Length == 0)
    {
        throw new ArgumentException("Array cannot be empty.");
    }
    T max = array[0];
    foreach (T item in array)
    {
        if (item.CompareTo(max) > 0)
        {
            max = item;
        }
    }
    return max;
}
// Usage
int[] numbers = { 1, 5, 2, 9, 3 };
int maxNumber = FindMax(numbers); // Output: 9
In this example, FindMax<T> is a generic method that finds the maximum value in an array of any data type that implements the IComparable interface.
3. Generic Interfaces
public interface IRepository<T>
{
    void Add(T item);
    void Delete(T item);
    T GetById(int id);
}
public class UserRepository : IRepository<User>
{
    public void Add(User item)
    {
        // Implementation
    }
    public void Delete(User item)
    {
        // Implementation
    }
    public User GetById(int id)
    {
        // Implementation
        return null;
    }
}
In this example, IRepository<T> is a generic interface defining basic operations for a repository. The UserRepository class implements this interface for the User type.
Conclusion
Generics in C# provide a powerful way to create flexible and type-safe code, making your applications more maintainable and efficient. By leveraging generics, you can design components that can adapt to different data types while ensuring type safety at compile-time. This results in cleaner, more maintainable code that can be easily reused across various parts of your application.