🚀 快速安装
复制以下命令并运行,立即安装此 Skill:
npx @anthropic-ai/skills install wshobson/agents/architecture-patterns
💡 提示:需要 Node.js 和 NPM
架构模式
掌握经过验证的后端架构模式,包括整洁架构、六边形架构和领域驱动设计,以构建可维护、可测试和可扩展的系统。
何时使用此技能
- 从头设计新的后端系统
- 重构单体应用以提高可维护性
- 为你的团队建立架构标准
- 从紧密耦合架构迁移到松散耦合架构
- 实现领域驱动设计原则
- 创建可测试和可模拟的代码库
- 规划微服务拆分
核心概念
1. 整洁架构(鲍勃大叔)
分层(依赖向内流动):
- 实体:核心业务模型
- 用例:应用程序业务规则
- 接口适配器:控制器、展示器、网关
- 框架与驱动:用户界面、数据库、外部服务
关键原则:
- 依赖指向内部
- 内层对外层一无所知
- 业务逻辑独立于框架
- 可在没有用户界面、数据库或外部服务的情况下进行测试
2. 六边形架构(端口与适配器)
组件:
- 领域核心:业务逻辑
- 端口:定义交互的接口
- 适配器:端口的实现(数据库、REST、消息队列)
优点:
- 易于切换实现(为测试提供模拟)
- 与技术无关的核心
- 关注点清晰分离
3. 领域驱动设计
战略模式:
- 限界上下文:不同领域的独立模型
- 上下文映射图:上下文之间的关系
- 通用语言:共享的术语
战术模式:
- 实体:具有标识的对象
- 值对象:由属性定义的不可变对象
- 聚合:一致性边界
- 仓储:数据访问抽象
- 领域事件:已发生的事情
整洁架构模式
目录结构
app/
├── domain/ # 实体与业务规则
│ ├── entities/
│ │ ├── user.py
│ │ └── order.py
│ ├── value_objects/
│ │ ├── email.py
│ │ └── money.py
│ └── interfaces/ # 抽象接口
│ ├── user_repository.py
│ └── payment_gateway.py
├── use_cases/ # 应用程序业务规则
│ ├── create_user.py
│ ├── process_order.py
│ └── send_notification.py
├── adapters/ # 接口实现
│ ├── repositories/
│ │ ├── postgres_user_repository.py
│ │ └── redis_cache_repository.py
│ ├── controllers/
│ │ └── user_controller.py
│ └── gateways/
│ ├── stripe_payment_gateway.py
│ └── sendgrid_email_gateway.py
└── infrastructure/ # 框架与外部关注点
├── database.py
├── config.py
└── logging.py
实现示例
# domain/entities/user.py
from dataclasses import dataclass
from datetime import datetime
from typing import Optional
@dataclass
class User:
"""核心用户实体 - 无框架依赖。"""
id: str
email: str
name: str
created_at: datetime
is_active: bool = True
def deactivate(self):
"""业务规则:停用用户。"""
self.is_active = False
def can_place_order(self) -> bool:
"""业务规则:活跃用户可以下单。"""
return self.is_active
# domain/interfaces/user_repository.py
from abc import ABC, abstractmethod
from typing import Optional, List
from domain.entities.user import User
class IUserRepository(ABC):
"""端口:定义契约,无实现。"""
@abstractmethod
async def find_by_id(self, user_id: str) -> Optional[User]:
pass
@abstractmethod
async def find_by_email(self, email: str) -> Optional[User]:
pass
@abstractmethod
async def save(self, user: User) -> User:
pass
@abstractmethod
async def delete(self, user_id: str) -> bool:
pass
# use_cases/create_user.py
from domain.entities.user import User
from domain.interfaces.user_repository import IUserRepository
from dataclasses import dataclass
from datetime import datetime
import uuid
@dataclass
class CreateUserRequest:
email: str
name: str
@dataclass
class CreateUserResponse:
user: User
success: bool
error: Optional[str] = None
class CreateUserUseCase:
"""用例:编排业务逻辑。"""
def __init__(self, user_repository: IUserRepository):
self.user_repository = user_repository
async def execute(self, request: CreateUserRequest) -> CreateUserResponse:
# 业务验证
existing = await self.user_repository.find_by_email(request.email)
if existing:
return CreateUserResponse(
user=None,
success=False,
error="邮箱已存在"
)
# 创建实体
user = User(
id=str(uuid.uuid4()),
email=request.email,
name=request.name,
created_at=datetime.now(),
is_active=True
)
# 持久化
saved_user = await self.user_repository.save(user)
return CreateUserResponse(
user=saved_user,
success=True
)
# adapters/repositories/postgres_user_repository.py
from domain.interfaces.user_repository import IUserRepository
from domain.entities.user import User
from typing import Optional
import asyncpg
class PostgresUserRepository(IUserRepository):
"""适配器:PostgreSQL 实现。"""
def __init__(self, pool: asyncpg.Pool):
self.pool = pool
async def find_by_id(self, user_id: str) -> Optional[User]:
async with self.pool.acquire() as conn:
row = await conn.fetchrow(
"SELECT * FROM users WHERE id = $1", user_id
)
return self._to_entity(row) if row else None
async def find_by_email(self, email: str) -> Optional[User]:
async with self.pool.acquire() as conn:
row = await conn.fetchrow(
"SELECT * FROM users WHERE email = $1", email
)
return self._to_entity(row) if row else None
async def save(self, user: User) -> User:
async with self.pool.acquire() as conn:
await conn.execute(
"""
INSERT INTO users (id, email, name, created_at, is_active)
VALUES ($1, $2, $3, $4, $5)
ON CONFLICT (id) DO UPDATE
SET email = $2, name = $3, is_active = $5
""",
user.id, user.email, user.name, user.created_at, user.is_active
)
return user
async def delete(self, user_id: str) -> bool:
async with self.pool.acquire() as conn:
result = await conn.execute(
"DELETE FROM users WHERE id = $1", user_id
)
return result == "DELETE 1"
def _to_entity(self, row) -> User:
"""将数据库行映射到实体。"""
return User(
id=row["id"],
email=row["email"],
name=row["name"],
created_at=row["created_at"],
is_active=row["is_active"]
)
# adapters/controllers/user_controller.py
from fastapi import APIRouter, Depends, HTTPException
from use_cases.create_user import CreateUserUseCase, CreateUserRequest
from pydantic import BaseModel
router = APIRouter()
class CreateUserDTO(BaseModel):
email: str
name: str
@router.post("/users")
async def create_user(
dto: CreateUserDTO,
use_case: CreateUserUseCase = Depends(get_create_user_use_case)
):
"""控制器:仅处理 HTTP 关注点。"""
request = CreateUserRequest(email=dto.email, name=dto.name)
response = await use_case.execute(request)
if not response.success:
raise HTTPException(status_code=400, detail=response.error)
return {"user": response.user}
六边形架构模式
# 核心领域(六边形中心)
class OrderService:
"""领域服务 - 无基础设施依赖。"""
def __init__(
self,
order_repository: OrderRepositoryPort,
payment_gateway: PaymentGatewayPort,
notification_service: NotificationPort
):
self.orders = order_repository
self.payments = payment_gateway
self.notifications = notification_service
async def place_order(self, order: Order) -> OrderResult:
# 业务逻辑
if not order.is_valid():
return OrderResult(success=False, error="订单无效")
# 使用端口(接口)
payment = await self.payments.charge(
amount=order.total,
customer=order.customer_id
)
if not payment.success:
return OrderResult(success=False, error="支付失败")
order.mark_as_paid()
saved_order = await self.orders.save(order)
await self.notifications.send(
to=order.customer_email,
subject="订单已确认",
body=f"订单 {order.id} 已确认"
)
return OrderResult(success=True, order=saved_order)
# 端口(接口)
class OrderRepositoryPort(ABC):
@abstractmethod
async def save(self, order: Order) -> Order:
pass
class PaymentGatewayPort(ABC):
@abstractmethod
async def charge(self, amount: Money, customer: str) -> PaymentResult:
pass
class NotificationPort(ABC):
@abstractmethod
async def send(self, to: str, subject: str, body: str):
pass
# 适配器(实现)
class StripePaymentAdapter(PaymentGatewayPort):
"""主适配器:连接到 Stripe 应用程序编程接口。"""
def __init__(self, api_key: str):
self.stripe = stripe
self.stripe.api_key = api_key
async def charge(self, amount: Money, customer: str) -> PaymentResult:
try:
charge = self.stripe.Charge.create(
amount=amount.cents,
currency=amount.currency,
customer=customer
)
return PaymentResult(success=True, transaction_id=charge.id)
except stripe.error.CardError as e:
return PaymentResult(success=False, error=str(e))
class MockPaymentAdapter(PaymentGatewayPort):
"""测试适配器:无外部依赖。"""
async def charge(self, amount: Money, customer: str) -> PaymentResult:
return PaymentResult(success=True, transaction_id="mock-123")
领域驱动设计模式
# 值对象(不可变)
from dataclasses import dataclass
from typing import Optional
@dataclass(frozen=True)
class Email:
"""值对象:验证过的邮箱。"""
value: str
def __post_init__(self):
if "@" not in self.value:
raise ValueError("无效的邮箱")
@dataclass(frozen=True)
class Money:
"""值对象:带货币的金额。"""
amount: int # 分
currency: str
def add(self, other: "Money") -> "Money":
if self.currency != other.currency:
raise ValueError("货币不匹配")
return Money(self.amount + other.amount, self.currency)
# 实体(具有标识)
class Order:
"""实体:具有标识,可变状态。"""
def __init__(self, id: str, customer: Customer):
self.id = id
self.customer = customer
self.items: List[OrderItem] = []
self.status = OrderStatus.PENDING
self._events: List[DomainEvent] = []
def add_item(self, product: Product, quantity: int):
"""实体中的业务逻辑。"""
item = OrderItem(product, quantity)
self.items.append(item)
self._events.append(ItemAddedEvent(self.id, item))
def total(self) -> Money:
"""计算属性。"""
return sum(item.subtotal() for item in self.items)
def submit(self):
"""带有业务规则的状态转换。"""
if not self.items:
raise ValueError("无法提交空订单")
if self.status != OrderStatus.PENDING:
raise ValueError("订单已提交")
self.status = OrderStatus.SUBMITTED
self._events.append(OrderSubmittedEvent(self.id))
# 聚合(一致性边界)
class Customer:
"""聚合根:控制对实体的访问。"""
def __init__(self, id: str, email: Email):
self.id = id
self.email = email
self._addresses: List[Address] = []
self._orders: List[str] = [] # 订单 ID,非完整对象
def add_address(self, address: Address):
"""聚合强制保持不变量。"""
if len(self._addresses) >= 5:
raise ValueError("最多允许 5 个地址")
self._addresses.append(address)
@property
def primary_address(self) -> Optional[Address]:
return next((a for a in self._addresses if a.is_primary), None)
# 领域事件
@dataclass
class OrderSubmittedEvent:
order_id: str
occurred_at: datetime = field(default_factory=datetime.now)
# 仓储(聚合持久化)
class OrderRepository:
"""仓储:持久化/检索聚合。"""
async def find_by_id(self, order_id: str) -> Optional[Order]:
"""从存储中重建聚合。"""
pass
async def save(self, order: Order):
"""持久化聚合并发布事件。"""
await self._persist(order)
await self._publish_events(order._events)
order._events.clear()
📄 原始文档
完整文档(英文):
https://skills.sh/wshobson/agents/architecture-patterns
💡 提示:点击上方链接查看 skills.sh 原始英文文档,方便对照翻译。
声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

评论(0)