معماری Clean (Clean Architecture)

Clean

معماری Clean یک روش ساختاردهی نرم‌افزار است که توسط رابرت سی. مارتین (Robert C. Martin) معروف به “عمو باب”) ارائه شد. این معماری بر اساس اصول SOLID و با هدف ایجاد سیستم‌های نرم‌افزاری قابل نگهداری، قابل تست و مستقل از جزئیات پیاده‌سازی طراحی شده است.

اصول کلیدی معماری Clean

  1. استقلال از فریمورک‌ها: سیستم به فریمورک‌های خاص وابسته نیست
  2. قابل تست بودن: کسب‌وکار منطق باید بدون UI، دیتابیس یا هر عنصر خارجی دیگر قابل تست باشد
  3. استقلال از UI: رابط کاربری می‌تواند تغییر کند بدون تأثیر بر لایه‌های دیگر
  4. استقلال از دیتابیس: می‌توان دیتابیس را تغییر داد بدون تأثیر بر لایه‌های کسب‌وکار
  5. استقلال از عوامل خارجی: منطق کسب‌وکار از دنیای خارج (APIها، سرویس‌های خارجی) بی‌خبر است

لایه‌های معماری Clean

1. لایه Entities (Enterprise Business Rules)

  • شامل مدل‌های اصلی کسب‌وکار و قوانین سازمان
  • مستقل از هر چیز دیگر در سیستم
  • مثال: کلاس Customer با خصوصیات و رفتارهای اصلی

2. لایه Use Cases (Application Business Rules)

  • شامل منطق خاص برنامه و گردش کارها
  • واسطه بین Entities و لایه‌های خارجی
  • مثال: RegisterCustomerUseCase که عملیات ثبت مشتری را مدیریت می‌کند

3. لایه Interface Adapters

  • تبدیل داده بین لایه Use Cases و دنیای خارج
  • شامل:
    • Controllers: مدیریت درخواست‌های HTTP
    • Presenters/ViewModels: آماده‌سازی داده برای نمایش
    • Gateways/Repositories: واسطه دسترسی به داده‌های خارجی

4. لایه Frameworks & Drivers (External Agencies)

  • شامل جزئیات پیاده‌سازی:
    • UI (وب، موبایل، دسکتاپ)
    • دیتابیس (SQL، NoSQL)
    • فریمورک‌ها (Spring, .NET, etc.)
    • کتابخانه‌های خارجی

قانون وابستگی (Dependency Rule)

وابستگی فقط باید به سمت داخل باشد:

  • لایه‌های بیرونی می‌توانند به لایه‌های درونی وابسته باشند
  • لایه‌های درونی هرگز نباید به لایه‌های بیرونی وابسته باشند
  • این قانون با استفاده از وارونگی وابستگی (Dependency Inversion) از SOLID اعمال می‌شود

مزایای معماری Clean

  1. قابلیت نگهداری بالا: تغییرات آسان‌تر می‌شوند
  2. قابلیت تست پذیری: تست واحد آسان‌تر می‌شود
  3. استقلال از فناوری: امکان تغییر تکنولوژی‌ها بدون تأثیر بر هسته سیستم
  4. قابلیت توسعه: افزودن ویژگی‌های جدید ساده‌تر می‌شود
  5. جداسازی واضح مسئولیت‌ها: هر لایه مسئولیت مشخصی دارد

معایب معماری Clean

  1. پیچیدگی اولیه: نیاز به طراحی دقیق دارد
  2. منحنی یادگیری: برای تیم‌های تازه کار ممکن است چالش‌برانگیز باشد
  3. بار اضافی: برای پروژه‌های کوچک ممکن است بیش از حد باشد

مثال:

// لایه Entities
public class Customer {
    private String id;
    private String name;
    // رفتارهای کسب‌وکار اصلی
}

// لایه Use Cases
public interface RegisterCustomerUseCase {
    Customer registerCustomer(String name);
}

// لایه Interface Adapters
@RestController
public class CustomerController {
    private final RegisterCustomerUseCase useCase;
    
    @PostMapping("/customers")
    public ResponseEntity<Customer> register(@RequestBody String name) {
        Customer customer = useCase.registerCustomer(name);
        return ResponseEntity.ok(customer);
    }
}

// لایه Frameworks & Drivers
@Repository
public class CustomerRepositoryImpl implements CustomerRepository {
    // پیاده‌سازی دسترسی به دیتابیس
}

تفاوت با سایر معماری‌ها

  • MVC: معماری Clean جداسازی بهتری ارائه می‌دهد و لایه Model را به چند لایه تقسیم می‌کند
  • Layered Architecture: معماری Clean وابستگی‌های بهتری با قانون وابستگی مدیریت می‌کند
  • Hexagonal: هر دو اهداف مشابهی دارند اما معماری Clean لایه‌بندی مشخص‌تری دارد

معماری Clean یک روش ساختاردهی نرم‌افزار است که با جداسازی قوی لایه‌ها و مدیریت وابستگی‌ها، سیستم‌های انعطاف‌پذیر و قابل نگهداری ایجاد می‌کند. این معماری به ویژه برای سیستم‌های پیچیده و بلندمدت مناسب است.

لایه‌های معماری تمیز

در معماری Clean (معماری تمیز)، سیستم به چندین لایه مجزا تقسیم می‌شود که هر کدام مسئولیت مشخصی دارند. این لایه‌ها به صورت دایره‌ای (مثل پیاز) قرار می‌گیرند و وابستگی فقط از لایه‌های بیرونی به سمت لایه‌های درونی مجاز است.

لایه‌های اصلی معماری Clean (از درونی‌ترین به بیرونی‌ترین):


1. لایه Entities (هسته مرکزی – Enterprise Business Rules)

  • مسئولیت: مدل‌سازی داده‌ها و قوانین اصلی کسب‌وکار
  • ویژگی‌ها:
    • مستقل از هر چیز خارجی (UI، دیتابیس، فریمورک‌ها)
    • شامل مدل‌های پایه و قوانین اساسی سازمان

مثال:

public class User {
    private String id;
    private String name;
    
    // قوانین کسب‌وکار (مثلاً اعتبارسنجی نام)
    public boolean isValidName() {
        return name != null && name.length() > 2;
    }
}

2. لایه Use Cases (Application Business Rules)

  • مسئولیت: پیاده‌سازی منطق خاص برنامه و سناریوهای کاربردی
  • ویژگی‌ها:
    • گردش کارهای برنامه را مدیریت می‌کند (مثلاً ثبت کاربر، پرداخت)
    • از Entities استفاده می‌کند اما از جزئیات بیرونی (مثل دیتابیس) بی‌خبر است

مثال

public class RegisterUserUseCase {
    private final UserRepository repository;
    
    public User execute(String name) {
        User user = new User(name);
        if (user.isValidName()) {
            return repository.save(user);
        }
        throw new InvalidUserException();
    }
}

3. لایه Interface Adapters (پل ارتباطی)

  • مسئولیت: تبدیل داده بین لایه Use Cases و دنیای خارج (مثل UI، دیتابیس، APIها)
  • زیرلایه‌ها:
    • Controllers: مدیریت درخواست‌های HTTP (در برنامه‌های وب)
    • Presenters/ViewModels: آماده‌سازی داده برای نمایش در UI
    • Gateways/Repositories: واسط برای دسترسی به داده‌های خارجی (مثل دیتابیس)

مثال

@RestController
public class UserController {
    private final RegisterUserUseCase useCase;
    
    @PostMapping("/users")
    public ResponseEntity&lt;User> register(@RequestBody String name) {
        User user = useCase.execute(name);
        return ResponseEntity.ok(user);
    }
}

4. لایه Frameworks & Drivers (بیرونی‌ترین لایه)

  • مسئولیت: پیاده‌سازی جزئیات فنی و ابزارهای خارجی
  • اجزاء:
    • UI: فرانت‌اند (React, Angular, Android, etc.)
    • دیتابیس: MySQL, MongoDB, etc.
    • فریمورک‌ها: Spring, Hibernate, etc.
    • کتابخانه‌های خارجی: مثلاً برای ارسال ایمیل

مثال

@Repository
public class UserRepositoryImpl implements UserRepository {
    private final JpaRepository jpaRepo;
    
    @Override
    public User save(User user) {
        // تبدیل User به Entity دیتابیس و ذخیره آن
        return jpaRepo.save(user);
    }
}

نمودار وابستگی لایه‌ها:

Frameworks &amp; Drivers (بیرون)  
       ↑  
Interface Adapters  
       ↑  
Use Cases  
       ↑  
Entities (داخل)

قانون طلایی وابستگی:

  • وابستگی فقط به سمت داخل است:
    • لایه‌های بیرونی می‌توانند به لایه‌های درونی وابسته باشند.
    • لایه‌های درونی هرگز نباید به لایه‌های بیرونی وابسته باشند.
    • این قانون با Dependency Injection و اینترفیس‌ها پیاده‌سازی می‌شود.

مثال ساده وابستگی:

// Use Case به Repository وابسته است، اما از طریق اینترفیس:
public class RegisterUserUseCase {
    private final UserRepository repository; // نه UserRepositoryImpl!
}

این لایه‌بندی باعث می‌شود تغییر در UI یا دیتابیس، هسته اصلی برنامه (Entities و Use Cases) را تحت تأثیر قرار ندهد.

اگر لایه‌بندی پروژه را به شکل چهارلایه اصلی (Domain, Infrastructure, Application, Presentation) طراحی کنیم، ساختار و مسئولیت‌های هر لایه به شرح زیر خواهد بود:


1. لایه Domain (هسته مرکزی)

مسئولیت‌ها:

  • تعریف موجودیت‌ها (Entities)، Value Objects و قوانین کسب‌وکار خالص.
  • شامل اینترفیس‌های ریپازیتوری (مثلاً IAccountRepository) برای جلوگیری از وابستگی به پیاده‌سازی.
  • مستقل کامل از فناوری‌های بیرونی (UI، دیتابیس، فریمورک‌ها).

اجزاء کلیدی:

  • موجودیت‌های اصلی مانند AccountJournalEntryTransaction.
  • Enumها مانند AccountType (دارایی، بدهی، …).
  • استثناهای سفارشی (DomainException).

مثال موجودیت Account:

public class Account : Entity<Guid> {
    public string Code { get; private set; } // مثال: 101.01
    public string Name { get; private set; }
    public AccountType Type { get; private set; }
    public AccountLevel Level { get; private set; } // کل، معین، تفصیلی

    // قوانین کسب‌وکار:
    public void UpdateName(string newName) {
        if (string.IsNullOrEmpty(newName))
            throw new DomainException("نام حساب نمی‌تواند خالی باشد.");
        Name = newName;
    }
}

2. لایه Infrastructure (پیاده‌سازی فنی)

مسئولیت‌ها:

  • پیاده‌سازی واسط‌های تعریف‌شده در لایه Domain (مثل IAccountRepository).
  • ارتباط با دیتابیس (EF Core, Dapper), فایل‌سیستمسرویس‌های خارجی (API, SMS, Email).
  • مدیریت کش (Cache)لاگ‌گیری, و احراز هویت.

اجزاء کلیدی:

  • ریپازیتوری‌ها (مثلاً AccountRepository).
  • DbContext و Migrationها.
  • سرویس‌های инфраструктуры مانند EmailService.

مثال ریپازیتوری:

public class AccountRepository : IAccountRepository {
    private readonly AccountingDbContext _context;

    public AccountRepository(AccountingDbContext context) {
        _context = context;
    }

    public async Task AddAsync(Account account) {
        await _context.Accounts.AddAsync(account);
        await _context.SaveChangesAsync();
    }
}

3. لایه Application (منطق کاربرد)

مسئولیت‌ها:

  • پیاده‌سازی Use Cases (مثل ثبت سند حسابداری، محاسبه ترازنامه).
  • تعریف DTOها برای انتقال داده بین لایه‌ها.
  • اعتبارسنجی ورودی‌ها (با FluentValidation یا DataAnnotations).

اجزاء کلیدی:

  • Handlers (پردازش Commands و Queries).
  • Services (منطق چندوجهی مانند TaxCalculator).
  • Mappers (تبدیل Entity به DTO).

مثال Use Case:

public class CreateAccountCommandHandler : IRequestHandler&lt;CreateAccountCommand, Guid> {
    private readonly IAccountRepository _repository;

    public CreateAccountCommandHandler(IAccountRepository repository) {
        _repository = repository;
    }

    public async Task&lt;Guid> Handle(CreateAccountCommand command) {
        var account = new Account(command.Code, command.Name, command.Type);
        await _repository.AddAsync(account);
        return account.Id;
    }
}

4. لایه Presentation (رابط کاربری)

مسئولیت‌ها:

  • نمایش داده‌ها و دریافت ورودی از کاربر.
  • وابسته به پلتفرم (مثلاً WinForms, ASP.NET Core, WPF).

اجزاء کلیدی:

  • Controllers (در وب) یا ViewModels (در WPF).
  • Razor Pages یا Blazor Components.
  • API Endpoints (در حالت Web API).

مثال Controller در ASP.NET Core:

[ApiController]
[Route("api/accounts")]
public class AccountController : ControllerBase {
    private readonly IMediator _mediator;

    public AccountController(IMediator mediator) {
        _mediator = mediator;
    }

    [HttpPost]
    public async Task&lt;IActionResult> Create([FromBody] CreateAccountCommand command) {
        var accountId = await _mediator.Send(command);
        return Ok(accountId);
    }
}

نمودار وابستگی‌ها:

Presentation (بیرونی‌ترین لایه)

Application

Domain ← Infrastructure

قانون وابستگی:

  • لایه‌های بالایی فقط به لایه‌های پایینی وابسته‌اند:
    • Presentation → Application → Domain.
    • Infrastructure → Domain (پیاده‌سازی اینترفیس‌های دامنه).
  • هرگز برعکس (مثلاً Domain نباید به Infrastructure وابسته باشد).

مقایسه با معماری Clean کلاسیک:

معماری چهارلایهمعماری Clean
Presentation = Frameworks & DriversPresentation = UI
Application = Use CasesApplication = Use Cases + Interface Adapters
Infrastructure = Interface Adapters + FrameworksInfrastructure = جدا شده به لایه‌های دقیق‌تر
Domain = EntitiesDomain = Entities + Enterprise Rules

مزایای این ساختار:

  1. سادگی: تقسیم‌بندی واضح برای پروژه‌های متوسط.
  2. انعطاف‌پذیری: امکان جایگزینی آسان Infrastructure (مثلاً تغییر از EF Core به Dapper).
  3. تست‌پذیری: تست Application بدون نیاز به Presentation یا Infrastructure.

معایب:

  • عدم تفکیک دقیق بین Interface Adapters و Frameworks (مشکل در پروژه‌های بسیار بزرگ).
  • Application ممکن است بیش از حد سنگین شود (در صورت عدم تقسیم به ماژول‌های کوچک‌تر).

جمع‌بندی:

این معماری برای اکثر پروژه‌های تجاری (خصوصاً با چارچوب‌هایی مانند ASP.NET Core یا WPF) مناسب است. اگر پروژه بسیار پیچیده باشد (مثل سیستم‌های بانکی)، استفاده از معماری Clean با لایه‌های دقیق‌تر (EntitiesUse CasesInterface AdaptersFrameworks) توصیه می‌شود.

(DomainInfrastructureApplicationPresentation) ارائه می‌شود. این نمودار شامل پوشه‌های داخلی، زیرلایه‌ها و الگوهای معماری مرتبط است:

کلیدواژه‌های معماری و الگوها:

لایهالگوها/اصولتوضیح
DomainDDD (Domain-Driven Design)طراحی بر اساس مدل‌های کسب‌وکار
Entity Patternموجودیت‌ها با شناسه یکتا
Value Object Patternاشیاء بدون شناسه (مثل Money)
InfrastructureRepository Patternجداسازی دسترسی به داده
Unit of Workمدیریت تراکنش‌های دیتابیس
Dependency Injectionتزریق وابستگی‌ها
ApplicationCQRS (Command-Query Separation)تفکیک عملیات نوشتن و خواندن
Mediator Patternکاهش وابستگی با واسطه (MediatR)
Fluent Validationاعتبارسنجی declarative
PresentationMVC/MVVMجداسازی منطق نمایش
REST APIبرای سرویس‌های وب
Clean UIحداقل منطق در لایه نمایش
deepseek mermaid 20250705 533f95
  • پیکان‌ها: جهت وابستگی (مثلاً Presentation فقط به Application وابسته است).
  • رنگ‌ها:
    • بنفش: لایه نمایش
    • آبی: منطق کاربرد
    • نارنجی: دامنه
    • سبز: زیرساخت

نکات کلیدی:

  1. Domain نباید به هیچ لایه دیگری وابسته باشد.
  2. Infrastructure می‌تواند به Domain وابسته باشد (برای پیاده‌سازی اینترفیس‌ها).
  3. الگوی CQRS در لایه Application باعث تمایز واضح بین عملیات نوشتن (Commands) و خواندن (Queries) می‌شود.
  4. Repository Pattern در Infrastructure از دامنه در برابر تغییرات دیتابیس محافظت می‌کند.

دیدگاه شما

نشانی ایمیل شما منتشر نخواهد شد.