.NET/C#의 코딩 규약, 명명 규칙, 레이아웃, C# 12/13/14의 최신 기능 활용 가이드라인을 정의합니다. C#/.NET 코드 작성 시, 클래스·메서드 명명 시, 코드 포맷팅 시, 또는 사용자가 코딩 규약, 명명 규칙, C# 모범 사례, Primary Constructors, Collection Expressions, field 키워드에 대해 언급했을 때 사용합니다.
/plugin marketplace add icartsh/icartsh_plugin/plugin install icartsh-plugin@icartsh-marketplaceThis skill inherits all available tools. When active, it can use any tool Claude has access to.
이 SKILL은 개발되는 모든 .NET 프로젝트에 적용되는 코딩 규약을 정의합니다. .NET 8 이후의 최신 기능(C# 12/13/14, .NET 8/9/10)을 적극적으로 활용하여 가독성, 유지보수성, 성능이 높은 코드를 구현하는 것을 목적으로 합니다.
이 SKILL은 다음 범위를 다룹니다:
_field) 사용은 절대 금지한다.NET 8 이후 각 버전에서 도입된 주요 기능을 적극적으로 활용합니다. 이전 버전과의 호환성이 필요한 경우를 제외하고 항상 최신 기능을 우선적으로 사용합니다.
좋은 예:
public class Person(string name, int age)
{
public string Name => name;
public int Age => age;
public void Display()
{
Console.WriteLine($"{name} is {age} years old");
}
}
나쁜 예:
public class Person
{
private string name;
private int age;
public Person(string name, int age)
{
this.name = name;
this.age = age;
}
public string Name => name;
public int Age => age;
}
좋은 예:
int[] array = [1, 2, 3, 4, 5];
List<string> list = ["one", "two", "three"];
int[] row0 = [1, 2, 3];
int[] row1 = [4, 5, 6];
// 스프레드 연산자로 결합
int[] combined = [..row0, ..row1];
좋은 예:
var incrementBy = (int source, int increment = 1) => source + increment;
Console.WriteLine(incrementBy(5));
Console.WriteLine(incrementBy(5, 3));
좋은 예:
using Point = (int x, int y);
using ProductList = System.Collections.Generic.List<(string Name, decimal Price)>;
Point origin = (0, 0);
ProductList products = [("Product1", 100m), ("Product2", 200m)];
params 수식어를 배열 외의 컬렉션 타입에서도 사용 가능해짐List<T>, Span<T>, ReadOnlySpan<T>, IEnumerable<T> 등에서 사용 가능좋은 예:
public void ProcessItems(params List<string> items)
{
foreach (var item in items)
{
Console.WriteLine(item);
}
}
// 메모리 효율이 중요한 경우
public void ProcessData(params ReadOnlySpan<int> data)
{
foreach (var value in data)
{
Process(value);
}
}
System.Threading.Lock 타입을 사용하여 더 빠른 스레드 동기화를 구현함Monitor 기반 락보다 빠름좋은 예:
private readonly Lock lockObject = new();
public void UpdateData()
{
lock (lockObject)
{
// Critical section
}
}
나쁜 예:
// 기존 object 기반 락 (C# 13에서는 권장되지 않음)
private readonly object lockObject = new();
public void UpdateData()
{
lock (lockObject)
{
// Critical section
}
}
좋은 예:
// 정의 부분
public partial class DataModel
{
public partial string Name { get; set; }
}
// 구현 부분
public partial class DataModel
{
private string name;
public partial string Name
{
get => name;
set => name = value ?? throw new ArgumentNullException(nameof(value));
}
}
^ 연산자를 사용할 수 있게 됨좋은 예:
var countdown = new TimerBuffer
{
buffer =
{
[^1] = 0,
[^2] = 1,
[^3] = 2
}
};
ref struct 타입이 인터페이스를 구현할 수 있게 됨ref struct를 사용할 수 있게 됨 (allows ref struct 제약 조건)좋은 예:
public ref struct SpanWrapper<T> : IEnumerable<T>
{
private Span<T> span;
public IEnumerator<T> GetEnumerator()
{
foreach (var item in span)
{
yield return item;
}
}
}
좋은 예:
extension<TSource>(IEnumerable<TSource> source)
{
public bool IsEmpty => !source.Any();
public int Count => source.Count();
}
field 키워드를 사용하여 명시적인 backing field를 줄임좋은 예:
// C# 14의 field 키워드 사용
public string Name
{
get;
set => field = value ?? throw new ArgumentNullException(nameof(value));
}
// 어쩔 수 없이 명시적인 backing field가 필요한 경우에도 언더스코어 없음
private string name;
public string Name
{
get => name;
set => name = value ?? throw new ArgumentNullException(nameof(value));
}
나쁜 예:
// 언더스코어 접두사는 절대 금지
private string _name;
public string Name
{
get => _name;
set => _name = value ?? throw new ArgumentNullException(nameof(value));
}
?.를 사용하여 null 체크를 간결하게 기술함좋은 예:
customer?.Order = GetCurrentOrder();
나쁜 예:
if (customer != null)
{
customer.Order = GetCurrentOrder();
}
Span<T>와 ReadOnlySpan<T>를 활용함좋은 예:
public class CustomerOrder
{
public string OrderId { get; set; }
public void ProcessOrder() { }
}
좋은 예:
public class OrderProcessor
{
// 언더스코어 없음
private string customerName;
// 언더스코어 없음
private int orderCount;
public void ProcessOrder(string orderId)
{
var customerName = GetCustomerName(orderId);
string processedResult = Process(customerName);
}
}
나쁜 예:
public class OrderProcessor
{
// 언더스코어 접두사는 절대 금지
private string _customerName;
// 언더스코어 접두사는 절대 금지
private int _orderCount;
}
I를 사용함좋은 예:
public interface IOrderProcessor
{
void Process(Order order);
}
T를 사용함좋은 예:
public class Repository<TEntity> where TEntity : class
{
public void Add(TEntity entity) { }
}
좋은 예:
public void ProcessOrder(Order order)
{
if (order != null)
{
order.Process();
}
}
// 1행이라도 중괄호를 사용함
if (isValid)
{
Execute();
}
for (int i = 0; i < 10; i++)
{
Process(i);
}
나쁜 예:
// 중괄호 생략 금지
if (isValid)
Execute();
// 중괄호 생략 금지
for (int i = 0; i < 10; i++)
Process(i);
// 중괄호 생략 금지
if (order != null) order.Process();
좋은 예:
public class Order
{
public string OrderId { get; set; }
public void Process()
{
var result = Validate();
Execute(result);
}
private bool Validate()
{
return OrderId != null;
}
}
좋은 예:
namespace YourProject.Orders;
public class OrderProcessor
{
// Implementation
}
나쁜 예:
namespace YourProject.Orders
{
public class OrderProcessor
{
// Implementation
}
}
좋은 예:
using System;
using System.Collections.Generic;
using System.Linq;
namespace YourProject.Orders;
string, int, bool)를 사용함System.String, System.Int32)은 사용하지 않음좋은 예:
string name = "John";
int count = 10;
bool isValid = true;
나쁜 예:
String name = "John";
Int32 count = 10;
Boolean isValid = true;
var를 사용함좋은 예:
// 명백함
var orders = new List<Order>();
// 명백함
var customer = GetCustomer();
// 기본 타입은 명시
int count = 10;
// 기본 타입은 명시
string name = "John";
나쁜 예:
// 기본 타입에서 var는 피함
var count = 10;
// 기본 타입에서 var는 피함
var name = "John";
좋은 예:
string message = $"Order {orderId} processed successfully";
나쁜 예:
string message = "Order " + orderId + " processed successfully";
StringBuilder를 사용함좋은 예:
var builder = new StringBuilder();
for (int i = 0; i < 1000; i++)
{
builder.Append($"Line {i}\n");
}
좋은 예:
string json = """
{
"name": "John",
"age": 30
}
""";
좋은 예:
var customer = new Customer
{
Name = "John",
Email = "john@example.com"
};
System.Exception 대신 구체적인 예외를 캐치함좋은 예:
try
{
ProcessOrder(order);
}
catch (ArgumentNullException ex)
{
Logger.Error("Order is null", ex);
}
나쁜 예:
try
{
ProcessOrder(order);
}
catch (Exception ex) // 너무 일반적임
{
Logger.Error("Error", ex);
}
using 문을 사용함좋은 예:
using var connection = new SqlConnection(connectionString);
connection.Open();
// Process
나쁜 예:
SqlConnection connection = null;
try
{
connection = new SqlConnection(connectionString);
connection.Open();
// Process
}
finally
{
connection?.Dispose();
}
좋은 예:
var activeCustomers = from customer in customers
where customer.IsActive
select customer;
where 절을 사용하여 조기에 데이터를 필터링함좋은 예:
var result = customers
.Where(c => c.IsActive)
.Select(c => c.Name)
.ToList();
좋은 예:
var query = from customer in customers
where customer.IsActive
select customer;
좋은 예:
button.Click += (s, e) => ProcessClick();
좋은 예:
TryParse<int> parse = (text, out result) => int.TryParse(text, out result);
//를 사용함좋은 예:
// 고객 주문을 처리함
ProcessOrder(order);
var processor = new OrderProcessor();
// 주문을 실행함
var result = processor.ProcessOrder(order);
나쁜 예:
ProcessOrder(order); // 고객 주문을 처리함 (코드와 같은 행은 금지)
var processor = new OrderProcessor();
// 이 행의 앞에 빈 행이 없음 (나쁜 예)
var result = processor.ProcessOrder(order);
좋은 예:
/// <summary>
/// 지정된 주문을 처리함
/// </summary>
/// <param name="order">처리할 주문</param>
/// <returns>처리 결과</returns>
public bool ProcessOrder(Order order)
{
// Implementation
}
좋은 예:
var result = OrderProcessor.ProcessOrder(order);
나쁜 예:
var processor = new OrderProcessor();
// 정적 메서드를 인스턴스를 통해 호출하는 것은 오해의 소지가 있음
var result = processor.ProcessOrder(order);
필수 규칙:
C# 12 이후 기능:
C# 13 이후 기능:
C# 14 이후 기능:
field 키워드를 사용하여 backing field를 간결하게 기술했음기본 규칙:
string, int)를 사용하고 있음var를 적절히 사용하고 있음 (타입이 명백한 경우에만)using 문을 사용하고 있음