اصول SOLID: راهنمای کامل طراحی نرمافزار
اصول SOLID چیست؟ این سوال بسیاری از توسعهدهندگان نرمافزار و مهندسان کیفی است. اصول SOLID مجموعهای از پنج قانون بنیادین برای طراحی کد تمیز، منعطف و قابل نگهداری هستند. بنابراین، در این مقاله همه نکات، مثالها و کاربردهای این اصول را بهتفصیل بررسی میکنیم تا شما بتوانید در پروژههای واقعی از آنها استفاده کنید.
تاریخچه و معرفی اصول SOLID
اصطلاح SOLID در دهه 2000 توسط رابرت مارتین (Robert C. Martin) شهرت یافت. اولین بار مایکل فِیشر (Michael Feathers) آن را معرفی کرد و سپس رابرت مارتین هر حرف را توضیح داد. بنابراین، این اصلاحات بهویژه در توسعه شیءگرا اهمیت زیادی پیدا کردند.
چرا اصول SOLID مهم است؟
-
باعث ایجاد کد پایدار، قابل توسعه و آسان برای نگهداری میشود.
-
کاهش هزینههای توسعه در بلندمدت.
-
باعث فهم بهتر و افزایش همکاری در تیمهای بزرگ میشود.
علاوه بر این، رعایت SOLID میتواند بهراحتی باعث شود تست واحد (Unit Testing) سادهتر و سریعتر انجام شود.
بررسی اصول SOLID
اصل تکمسئولیت (SRP)
-
هر کلاس باید یک مسئولیت واحد داشته باشد.
-
بنابراین، تغییرات در یک بُعد نباید به کلاسهای دیگر سرایت کند.
مثال:
کلاس Invoice هم حسابرسی و هم ارسال ایمیل را انجام ندهد. بهجای آن، دو کلاس جداگانه برای صدور و ارسال تعریف کنید.
// مسئول صدور فاکتور
public class InvoiceGenerator
{
public void GenerateInvoice(Order order)
{
// منطق تولید فاکتور
}
}
// مسئول ارسال فاکتور
public class InvoiceEmailSender
{
public void SendInvoiceEmail(Order order)
{
// منطق ارسال ایمیل
}
}
✅ InvoiceGenerator
و InvoiceEmailSender
هرکدام یک مسئولیت خاص دارند. تغییر در ایمیل هیچ ربطی به صدور فاکتور ندارد.
اصل باز-بسته (OCP)
-
سیستم باید برای توسعه باز و برای تغییر بسته باشد.
-
یعنی رفتار جدید با اضافه کردن کد امکانپذیر باشد، بدون تغییر در کد موجود.
روش اعمال:
استفاده از رابطها (interfaces) یا وراثت (inheritance) برای افزودن قابلیت بدون تغییر کد قبلی.
اصل باز-بسته (OCP)
public interface IDiscountCalculator
{
decimal CalculateDiscount(Order order);
}
public class RegularCustomerDiscount : IDiscountCalculator
{
public decimal CalculateDiscount(Order order) => order.Total * 0.05m;
}
public class PremiumCustomerDiscount : IDiscountCalculator
{
public decimal CalculateDiscount(Order order) => order.Total * 0.10m;
}
public class OrderProcessor
{
private readonly IDiscountCalculator _discountCalculator;
public OrderProcessor(IDiscountCalculator discountCalculator)
{
_discountCalculator = discountCalculator;
}
public decimal FinalPrice(Order order)
{
var discount = _discountCalculator.CalculateDiscount(order);
return order.Total - discount;
}
}
✅ بدون تغییر در OrderProcessor
میتوان تخفیفهای جدید اضافه کرد (Open for extension, closed for modification).
اصل جایگزینی لیسکف (LSP)
-
کلاسهای فرزند باید بتوانند بدون مشکل جایگزین والد شوند.
-
بنابراین، ویژگیهای اصلی را حفظ کرده و تغییر رفتار ندهند.
public abstract class Bird
{
public abstract void Fly();
}
public class Sparrow : Bird
{
public override void Fly()
{
Console.WriteLine("Sparrow is flying...");
}
}
// ❌ اینجا نقض LSP: پنگوئن نمیتونه پرواز کنه
public class Penguin : Bird
{
public override void Fly()
{
throw new NotSupportedException("Penguins can't fly!");
}
}
✅ راهحل: از یک Interface مناسبتر استفاده کنیم:
public interface IFlyable
{
void Fly();
}
public class Sparrow : IFlyable
{
public void Fly() => Console.WriteLine("Sparrow flying...");
}
public class Penguin
{
public void Swim() => Console.WriteLine("Penguin swimming...");
}
اصل جداسازی رابط (ISP)
-
استفاده از رابطهای کوچک و خاص بهجای یک رابط بزرگ.
-
بنابراین، کلاسها تابعهای غیرضروری دریافت نمیکنند.
مثال:
Interface برای چاپگر نباید تابع اسکن را داشته باشد؛ چنین رابطی باید جداگانه تعریف شود.
public interface IPrinter
{
void Print(Document doc);
void Scan(Document doc);
void Fax(Document doc);
}
❌ کلاس سادهای مثل SimplePrinter
نیاز به Fax نداره و مجبوره تابع خالی داشته باشه.
✅ راهحل:
public interface IPrintable
{
void Print(Document doc);
}
public interface IScannable
{
void Scan(Document doc);
}
حالا کلاسها فقط اون چیزی که لازم دارن رو پیادهسازی میکنن:
public class SimplePrinter : IPrintable
{
public void Print(Document doc) => Console.WriteLine("Printed");
}
اصل وارونگی وابستگی (DIP)
-
ماژولهای سطح بالا نباید به ماژولهای سطح پایین وابسته باشند.
-این وابستگی باید بهسمت انتزاعها (interfaces) برود.
public interface IMessageSender
{
void SendMessage(string message);
}
public class EmailSender : IMessageSender
{
public void SendMessage(string message)
{
Console.WriteLine($"Sending Email: {message}");
}
}
public class NotificationService
{
private readonly IMessageSender _sender;
public NotificationService(IMessageSender sender)
{
_sender = sender;
}
public void Notify(string message)
{
_sender.SendMessage(message);
}
}
✅ کلاس NotificationService به انتزاع وابسته است نه پیادهسازی. بنابراین میتونه هر نوع پیامرسانی رو قبول کنه (SMS, Slack, Telegram و …)
مثال عملی و مطالعه موردی
مثال ساده در PHP
interface PaymentGatewayInterface {
public function pay(int $amount): bool;
}
class StripeGateway implements PaymentGatewayInterface {
public function pay(int $amount): bool { /* اجرای پرداخت */ }
}
class OrderProcessor {
private $gateway;
public function __construct(PaymentGatewayInterface $gateway) {
$this->gateway = $gateway;
}
public function process(int $orderId): bool {
return $this->gateway->pay(100);
}
}
-
DIP رعایت شده چون
OrderProcessor
بهPaymentGatewayInterface
وابسته است نه StripeGateay. -
اضافه کردن درگاههای جدید نیازی به تغییر کلاس
OrderProcessor
ندارد (OCP).
مطالعه موردی واقعی
فرض کنید تیمی روی یک محصول بزرگ کار میکند. در گذشته، همه منطقها در یک کلاس قرار داشت. تغییر کوچک در گرافیک باعث خطا در منطقهای دیگر میشد.
با اعمال اصول SOLID:
-
کلاسها به ماژولهای تخصصی تبدیل شدند (SRP).
-
رابطها برای قسمتهای متغیر تعریف شد (ISP).
-
تزریق وابستگی (Dependency Injection) جایگزین نمونهسازی مستقیم شد.
نتیجه؟ ساختار قابل فهم، تستپذیر و نگهداری آسان.
مزایا و چالشها
مزایا
-
قابلیت نگهداری بالا
-
قابل تست بودن
-
توانایی توسعه سریعتر
چالشها
-
کمی پیچیدگی در طراحی ابتدایی
-
ممکن است وجود کلاسهای زیاد در پروژه
نکات کاربردی برای رعایت SOLID
-
شروع با SRP و DIP توصیه میشود.
-
استفاده از تزریق وابستگی (Dependency Injection)
-
تست واحد برای اطمینان از تعویضپذیری کلاسها
-
کد را بازنگری کنید و کلاسهای بزرگ را تقسیم کنید
-
همگامسازی با دیگر اعضای تیم
سؤالهای متداول (FAQs)
س: آیا SOLID فقط مخصوص زبانهای شیءگراست؟
پاسخ: این اصول بیشتر در طراحی شیءگرا کاربرد دارند، اما مفاهیم آنها را میتوان در سبکهای دیگر هم تحلیل کرد.
س: رعایت SOLID سرعت توسعه را کند میکند؟
نه. در طولانیمدت با کاهش خطا و تستپذیری، سرعت توسعه افزایش مییابد.
س: آیا همه اصول باید همزمان اعمال شوند؟
خیر. معمولاً با SRP و DIP شروع کنید، سپس بقیه را اضافه کنید.
منابع مرتبط
And To Do So From Now Until The Death, Whatever the Cost