C#JuniorTechnical
Что такое generics в C# и почему они полезны?
Generics в C# — параметризованные типы, проверяемые на этапе компиляции. Они устраняют boxing для value types, позволяют переиспользовать алгоритмы для любого типа и дают IntelliSense без приведений.
Что такое generics и зачем они нужны
Generics (обобщения) — механизм параметризации типов и методов в C#. Вместо написания одного класса для int и отдельного для string вы пишете один обобщённый тип Container<T>, а компилятор создаёт специализированные версии под каждый тип-аргумент.
Три главные причины использовать generics:
- Типобезопасность — ошибки типов обнаруживаются на этапе компиляции, не в runtime.
- Производительность — для value types (int, struct) CLR создаёт специализированный машинный код без boxing/unboxing.
- Переиспользование — один алгоритм работает с любым типом, удовлетворяющим ограничению (constraint).
Базовый пример: без generics vs с generics
// БЕЗ generics — до C# 2.0 (ArrayList)
var list = new System.Collections.ArrayList();
list.Add(42); // boxing int -> object
list.Add("mistake"); // компилируется, но это баг
int value = (int)list[0]; // явный cast, риск InvalidCastException
// С generics
var list = new List<int>();
list.Add(42); // нет boxing
// list.Add("mistake"); // ошибка компиляции CS1503
int value = list[0]; // нет cast
Generic методы и ограничения (constraints)
// Базовый generic метод
public T Max<T>(T a, T b) where T : IComparable<T>
=> a.CompareTo(b) >= 0 ? a : b;
// Использование
int maxInt = Max(3, 7); // T = int
string maxStr = Max("abc", "xyz"); // T = string
// Ограничения (constraints)
public class Repository<T> where T : class, IEntity, new()
{
// T : class — только reference types
// T : IEntity — T реализует интерфейс
// T : new() — T имеет публичный конструктор без параметров
public T Create() => new T();
public async Task SaveAsync(T entity, CancellationToken ct)
{
entity.UpdatedAt = DateTime.UtcNow;
await _db.Set<T>().AddAsync(entity, ct);
await _db.SaveChangesAsync(ct);
}
}
Generic классы и struct constraints для zero-allocation
// where T : struct — value type constraint, нет boxing
public readonly struct Option<T> where T : struct
{
private readonly T _value;
public bool HasValue { get; }
public Option(T value) { _value = value; HasValue = true; }
public T GetValueOrDefault(T @default = default)
=> HasValue ? _value : @default;
}
var opt = new Option<int>(42);
int val = opt.GetValueOrDefault(0); // нет heap allocation
Generics в стандартной библиотеке .NET
Generics пронизывают весь .NET:
List<T>,Dictionary<TKey, TValue>,HashSet<T>— коллекции.Task<TResult>,IAsyncEnumerable<T>— async pipeline.ILogger<T>,IOptions<T>— DI в ASP.NET Core.IEnumerable<T>,IQueryable<T>— LINQ и EF Core.Span<T>,Memory<T>,ReadOnlySpan<T>— zero-allocation работа с памятью.
Как CLR обрабатывает generics
CLR (Common Language Runtime) использует reification — физическое создание специализированных типов:
- Для value types:
List<int>иList<double>— разные JIT-скомпилированные типы, нет boxing. - Для reference types:
List<string>иList<User>используют одну JIT-специализацию с указателем, т.к. все reference types одного размера.
Подводные камни
- Constraint
where T : new()использует Activator.CreateInstance — медленнее прямогоnew T(); в .NET 8+ используйтеwhere T : IActivatableили фабрику. - Generic типы не поддерживают наследование по параметру:
List<Dog>не являетсяList<Animal>— для этого нужна ковариантность (out T) или контравариантность (in T) на интерфейсах. - Статические члены generic класса раздельны для каждого type argument:
Singleton<int>.Instance!=Singleton<string>.Instance. - Reflection с generics сложнее:
typeof(List<>)— открытый generic тип, требуетMakeGenericType(typeof(int))для создания экземпляра. - AOT (PublishAot) требует, чтобы все используемые generic инстанции были известны на этапе компиляции — динамические
MakeGenericTypeпроблематичны. - Слишком широкие constraints (
where T : class) снижают пользу generics — IDE не подсказывает члены T. - Generic методы в интерфейсах и default interface methods (.NET 8) — поведение при наследовании может быть неочевидным.
- Не путайте C# generics с Java generics: в Java type erasure убирает параметры типа в runtime, в C# — нет (reification).
Common mistakes
- Путать обобщения, типобезопасность и переиспользование с похожим механизмом из другой версии или платформы.
- Игнорировать runtime-границы C#: lifecycle, DI scope, SQL translation, UI thread или platform API.
- Не обсуждать null/empty/error cases и поведение под нагрузкой.
What the interviewer is testing
- Кандидат объясняет обобщения, типобезопасность и переиспользование на конкретном примере, а не только определением.
- Указывает последствия для производительности, тестируемости и поддержки.
- Различает документированное поведение текущего стека и устаревшие практики.