C# 12 - Quick Reference
Deep Knowledge: Use
mcp__documentation__fetch_docswith technology:csharpfor comprehensive documentation.
Records
// Positional record (immutable)
public record UserDto(int Id, string Name, string Email);
// Record with additional members
public record OrderDto(int Id, decimal Total)
{
public string FormattedTotal => Total.ToString("C");
}
// Record struct (value type, C# 10)
public readonly record struct Point(double X, double Y);
// With-expression (non-destructive mutation)
var updated = user with { Name = "New Name" };
Pattern Matching (C# 12)
// Switch expression
string GetDiscount(Customer customer) => customer switch
{
{ Type: CustomerType.Premium, YearsActive: > 5 } => "30%",
{ Type: CustomerType.Premium } => "20%",
{ Type: CustomerType.Regular, YearsActive: > 3 } => "10%",
_ => "0%",
};
// List patterns (C# 11)
var result = numbers switch
{
[1, 2, 3] => "exact match",
[1, .., 3] => "starts with 1, ends with 3",
[_, > 5, ..] => "second element > 5",
[] => "empty",
_ => "other",
};
// Type pattern with when
if (shape is Circle { Radius: > 10 } circle)
{
Console.WriteLine($"Large circle: {circle.Radius}");
}
Nullable Reference Types
// Enable in .csproj: <Nullable>enable</Nullable>
public class UserService
{
// Nullable return type
public User? FindById(int id) => _users.FirstOrDefault(u => u.Id == id);
// Non-nullable parameter
public void Update(User user)
{
ArgumentNullException.ThrowIfNull(user);
// user is guaranteed non-null here
}
// Null-conditional and coalescing
public string GetDisplayName(User? user)
=> user?.Name ?? "Unknown";
// Required members (C# 11)
public required string Name { get; init; }
}
Async/Await Patterns
// Basic async method
public async Task<User> GetUserAsync(int id)
{
var user = await _repository.FindAsync(id);
return user ?? throw new NotFoundException($"User {id} not found");
}
// Parallel async
public async Task<(User[], Order[])> GetDashboardDataAsync(int userId)
{
var usersTask = _userService.GetAllAsync();
var ordersTask = _orderService.GetByUserAsync(userId);
await Task.WhenAll(usersTask, ordersTask);
return (usersTask.Result, ordersTask.Result);
}
// IAsyncEnumerable
public async IAsyncEnumerable<User> GetUsersStreamAsync(
[EnumeratorCancellation] CancellationToken ct = default)
{
await foreach (var user in _context.Users.AsAsyncEnumerable().WithCancellation(ct))
{
yield return user;
}
}
// ValueTask for hot paths
public ValueTask<User?> GetCachedUserAsync(int id)
{
if (_cache.TryGetValue(id, out var user))
return ValueTask.FromResult<User?>(user);
return new ValueTask<User?>(LoadUserAsync(id));
}
LINQ
// Method syntax (preferred)
var result = users
.Where(u => u.IsActive)
.OrderBy(u => u.Name)
.Select(u => new { u.Name, u.Email })
.ToList();
// GroupBy
var grouped = orders
.GroupBy(o => o.Status)
.Select(g => new { Status = g.Key, Count = g.Count(), Total = g.Sum(o => o.Amount) });
// Aggregate
var summary = orders.Aggregate(
new { Count = 0, Total = 0m },
(acc, order) => new { Count = acc.Count + 1, Total = acc.Total + order.Amount });
// Chunk (C# 12 / .NET 8)
var batches = users.Chunk(100); // IEnumerable<User[]>
Primary Constructors (C# 12)
// Class primary constructor
public class UserService(IUserRepository repository, ILogger<UserService> logger)
{
public async Task<User> GetByIdAsync(int id)
{
logger.LogInformation("Getting user {Id}", id);
return await repository.GetByIdAsync(id)
?? throw new NotFoundException($"User {id} not found");
}
}
// Record primary constructor (already existed)
public record UserDto(int Id, string Name, string Email);
Collection Expressions (C# 12)
// Array
int[] numbers = [1, 2, 3, 4, 5];
// List
List<string> names = ["Alice", "Bob", "Charlie"];
// Spread
int[] first = [1, 2, 3];
int[] second = [4, 5, 6];
int[] combined = [..first, ..second]; // [1, 2, 3, 4, 5, 6]
Anti-Patterns
| Anti-Pattern | Why It's Bad | Correct Approach |
|--------------|--------------|------------------|
| .Result / .Wait() | Deadlocks | Use async/await |
| async void | Exceptions lost | Use async Task |
| Mutable DTOs | Unexpected mutations | Use record types |
| No null checking | NullReferenceException | Enable nullable references |
| String concatenation in loops | Memory pressure | Use StringBuilder |
Quick Troubleshooting
| Issue | Likely Cause | Solution |
|-------|--------------|----------|
| Nullable warning | Missing null check | Add ?, ??, or null guard |
| Deadlock | .Result in async context | Use await |
| LINQ multiple enumeration | Iterating IEnumerable twice | Call .ToList() first |
| Record equality fails | Reference type property | Override Equals or use value types |
微信扫一扫