docs/zh-CN/skills/hexagonal-architecture
stars:0
forks:0
watches:0
last updated:N/A
六边形架构
六边形架构(端口与适配器)使业务逻辑独立于框架、传输层和持久化细节。核心应用依赖于抽象端口,而适配器在边缘实现这些端口。
适用场景
- 构建需要长期可维护性和可测试性的新功能。
- 重构分层或框架密集型代码,其中领域逻辑与I/O关注点混杂。
- 为同一用例支持多种接口(HTTP、CLI、队列工作器、定时任务)。
- 替换基础设施(数据库、外部API、消息总线)而无需重写业务规则。
当需求涉及边界、领域驱动设计、重构紧耦合服务,或将应用逻辑与特定库解耦时,使用此技能。
核心概念
- 领域模型:业务规则和实体/值对象。无框架导入。
- 用例(应用层):编排领域行为和工作流步骤。
- 入站端口:描述应用能力的契约(命令/查询/用例接口)。
- 出站端口:应用所需依赖的契约(仓库、网关、事件发布器、时钟、UUID等)。
- 适配器:端口的基础设施和交付实现(HTTP控制器、数据库仓库、队列消费者、SDK封装器)。
- 组合根:将具体适配器绑定到用例的单一连接位置。
出站端口接口通常位于应用层(仅当抽象真正属于领域层时才位于领域层),而基础设施适配器实现它们。
依赖方向始终向内:
- 适配器 -> 应用/领域
- 应用 -> 端口接口(入站/出站契约)
- 领域 -> 仅领域抽象(无框架或基础设施依赖)
- 领域 -> 无外部依赖
工作原理
步骤1:建模用例边界
定义具有清晰输入和输出DTO的单个用例。将传输细节(Express req、GraphQL context、任务负载包装器)保持在此边界之外。
步骤2:首先定义出站端口
将每个副作用识别为端口:
- 持久化(
UserRepositoryPort) - 外部调用(
BillingGatewayPort) - 横切关注点(
LoggerPort、ClockPort)
端口应建模能力,而非技术。
步骤3:使用纯编排实现用例
用例类/函数通过构造函数/参数接收端口。它验证应用层不变量,协调领域规则,并返回纯数据结构。
步骤4:在边缘构建适配器
- 入站适配器将协议输入转换为用例输入。
- 出站适配器将应用契约映射到具体API/ORM/查询构建器。
- 映射保持在适配器中,而非用例内部。
步骤5:在组合根中连接所有组件
实例化适配器,然后将其注入用例。保持此连接集中化,以避免隐藏的服务定位器行为。
步骤6:按边界测试
- 使用伪造端口对用例进行单元测试。
- 使用真实基础设施依赖对适配器进行集成测试。
- 通过入站适配器对面向用户的流程进行端到端测试。
架构图
flowchart LR
Client["Client (HTTP/CLI/Worker)"] --> InboundAdapter["Inbound Adapter"]
InboundAdapter -->|"calls"| UseCase["UseCase (Application Layer)"]
UseCase -->|"uses"| OutboundPort["OutboundPort (Interface)"]
OutboundAdapter["Outbound Adapter"] -->|"implements"| OutboundPort
OutboundAdapter --> ExternalSystem["DB/API/Queue"]
UseCase --> DomainModel["DomainModel"]
建议的模块布局
使用以功能为先的组织方式,并带有显式边界:
src/
features/
orders/
domain/
Order.ts
OrderPolicy.ts
application/
ports/
inbound/
CreateOrder.ts
outbound/
OrderRepositoryPort.ts
PaymentGatewayPort.ts
use-cases/
CreateOrderUseCase.ts
adapters/
inbound/
http/
createOrderRoute.ts
outbound/
postgres/
PostgresOrderRepository.ts
stripe/
StripePaymentGateway.ts
composition/
ordersContainer.ts
TypeScript 示例
端口定义
export interface OrderRepositoryPort {
save(order: Order): Promise<void>;
findById(orderId: string): Promise<Order | null>;
}
export interface PaymentGatewayPort {
authorize(input: { orderId: string; amountCents: number }): Promise<{ authorizationId: string }>;
}
用例
type CreateOrderInput = {
orderId: string;
amountCents: number;
};
type CreateOrderOutput = {
orderId: string;
authorizationId: string;
};
export class CreateOrderUseCase {
constructor(
private readonly orderRepository: OrderRepositoryPort,
private readonly paymentGateway: PaymentGatewayPort
) {}
async execute(input: CreateOrderInput): Promise<CreateOrderOutput> {
const order = Order.create({ id: input.orderId, amountCents: input.amountCents });
const auth = await this.paymentGateway.authorize({
orderId: order.id,
amountCents: order.amountCents,
});
// markAuthorized returns a new Order instance; it does not mutate in place.
const authorizedOrder = order.markAuthorized(auth.authorizationId);
await this.orderRepository.save(authorizedOrder);
return {
orderId: order.id,
authorizationId: auth.authorizationId,
};
}
}
出站适配器
export class PostgresOrderRepository implements OrderRepositoryPort {
constructor(private readonly db: SqlClient) {}
async save(order: Order): Promise<void> {
await this.db.query(
"insert into orders (id, amount_cents, status, authorization_id) values ($1, $2, $3, $4)",
[order.id, order.amountCents, order.status, order.authorizationId]
);
}
async findById(orderId: string): Promise<Order | null> {
const row = await this.db.oneOrNone("select * from orders where id = $1", [orderId]);
return row ? Order.rehydrate(row) : null;
}
}
组合根
export const buildCreateOrderUseCase = (deps: { db: SqlClient; stripe: StripeClient }) => {
const orderRepository = new PostgresOrderRepository(deps.db);
const paymentGateway = new StripePaymentGateway(deps.stripe);
return new CreateOrderUseCase(orderRepository, paymentGateway);
};
多语言映射
在不同生态系统中使用相同的边界规则;仅语法和连接方式发生变化。
- TypeScript/JavaScript
- 端口:
application/ports/*作为接口/类型。 - 用例:带有构造函数/参数注入的类/函数。
- 适配器:
adapters/inbound/*、adapters/outbound/*。 - 组合:显式工厂/容器模块(无隐藏全局变量)。
- 端口:
- Java
- 包:
domain、application.port.in、application.port.out、application.usecase、adapter.in、adapter.out。 - 端口:
application.port.*中的接口。 - 用例:普通类(Spring
@Service是可选的,非必需)。 - 组合:Spring配置或手动连接类;将连接逻辑保持在领域/用例类之外。
- 包:
- Kotlin
- 模块/包镜像Java的拆分(
domain、application.port、application.usecase、adapter)。 - 端口:Kotlin接口。
- 用例:带有构造函数注入的类(Koin/Dagger/Spring/手动)。
- 组合:模块定义或专用组合函数;避免服务定位器模式。
- 模块/包镜像Java的拆分(
- Go
- 包:
internal/<feature>/domain、application、ports、adapters/inbound、adapters/outbound。 - 端口:由消费应用包拥有的小型接口。
- 用例:带有接口字段和显式
New...构造函数的结构体。 - 组合:在
cmd/<app>/main.go中连接(或专用连接包),保持构造函数显式。
- 包:
应避免的反模式
- 领域实体导入ORM模型、Web框架类型或SDK客户端。
- 用例直接从
req、res或队列元数据读取。 - 从用例直接返回数据库行,未经领域/应用映射。
- 让适配器直接相互调用,而非通过用例端口流转。
- 将依赖连接分散到多个文件中,使用隐藏的全局单例。
迁移手册
- 选择一个垂直切片(单个端点/任务),该切片频繁变更且带来痛苦。
- 提取具有显式输入/输出类型的用例边界。
- 围绕现有基础设施调用引入出站端口。
- 将编排逻辑从控制器/服务移动到用例中。
- 保留旧适配器,但使其委托给新用例。
- 围绕新边界添加测试(单元测试 + 适配器集成测试)。
- 逐个切片重复;避免完全重写。
重构现有系统
- 绞杀者模式:保留当前端点,一次将一个用例路由到新的端口/适配器。
- 无大爆炸式重写:按功能切片迁移,并通过特征化测试保持行为。
- 先建外观:在替换内部实现之前,将遗留服务包装在出站端口后面。
- 组合冻结:尽早集中连接,使新依赖不会泄漏到领域/用例层。
- 切片选择规则:优先处理高变更频率、低影响范围的流程。
- 回滚路径:为每个迁移的切片保留可逆开关或路由切换,直到生产行为得到验证。
测试指南(相同的六边形边界)
- 领域测试:将实体/值对象作为纯业务规则进行测试(无模拟,无框架设置)。
- 用例单元测试:使用出站端口的伪造/桩件测试编排;断言业务结果和端口交互。
- 出站适配器契约测试:在端口级别定义共享契约套件,并针对每个适配器实现运行。
- 入站适配器测试:验证协议映射(HTTP/CLI/队列负载到用例输入,以及输出/错误映射回协议)。
- 适配器集成测试:针对真实基础设施(数据库/API/队列)运行,测试序列化、模式/查询行为、重试和超时。
- 端到端测试:覆盖关键用户旅程,通过入站适配器 -> 用例 -> 出站适配器。
- 重构安全性:在提取之前添加特征化测试;保持它们直到新边界行为稳定且等价。
最佳实践清单
- 领域和应用层仅导入内部类型和端口。
- 每个外部依赖都由一个出站端口表示。
- 验证发生在边界处(入站适配器 + 用例不变量)。
- 使用不可变转换(返回新值/实体,而非修改共享状态)。
- 错误在边界间进行转换(基础设施错误 -> 应用/领域错误)。
- 组合根是显式的且易于审计。
- 用例可通过简单的内存伪造端口进行测试。
- 重构从具有行为保持测试的一个垂直切片开始。
- 语言/框架特定内容保持在适配器中,绝不进入领域规则。
