React 领域驱动设计:在 React 项目中划分领域逻辑层(Domain Layer)与 UI 呈现层

各位同学,大家下午好!

欢迎来到今天的讲座。我是你们的老朋友,一个在代码泥潭里摸爬滚打多年,头发比项目需求还少的资深编程专家。

今天我们要聊一个听起来很学术,但实际上能救你们狗命的话题——React 领域驱动设计(DDD):如何把你的 UI 层和那坨该死的业务逻辑分家。

我知道,你们心里可能在想:“专家,别扯那些虚头巴脑的理论了,我就想写个 React 组件,为什么非得搞什么 DDD?难道我以前写的代码是屎山,现在我要去盖摩天大楼吗?”

别急,先别急着把椅子扔向我。我们来聊聊为什么你的代码变成了“意大利面条”。

第一部分:当 React 遇上“上帝组件”

想象一下,你现在维护一个电商系统。有一天,老板说:“我要在这个购物车里加个功能,当总价超过 500 元时,自动打九折,并且给用户发个优惠券。”

你打开你的 CartPage.tsx,里面大概长这样:

import React, { useState, useEffect, useMemo } from 'react';
import axios from 'axios';

// 假设这是你的数据结构,全是 any,全是魔法
interface Item {
  id: number;
  price: number;
  quantity: number;
}

export const CartPage: React.FC = () => {
  const [items, setItems] = useState<Item[]>([]);
  const [discount, setDiscount] = useState(0);

  // 1. 获取数据
  useEffect(() => {
    axios.get('/api/cart').then(res => setItems(res.data));
  }, []);

  // 2. 业务逻辑混杂在 UI 逻辑里
  const calculateTotal = () => {
    let total = 0;
    items.forEach(item => {
      total += item.price * item.quantity;
    });

    // 业务规则:超过 500 打九折
    if (total > 500) {
      total = total * 0.9;
      // 业务规则:发优惠券
      setDiscount(10);
    } else {
      setDiscount(0);
    }
    return total;
  };

  // 3. UI 渲染
  return (
    <div className="cart-container">
      <h1>我的购物车</h1>
      <ul>
        {items.map(item => (
          <li key={item.id}>
            {item.name} - ${item.price * item.quantity}
          </li>
        ))}
      </ul>
      <div className="summary">
        总价: ${calculateTotal()}
        {discount > 0 && <span className="coupon">🎉 优惠券已发放!</span>}
      </div>
    </div>
  );
};

看,是不是很眼熟?这就是 90% 的 React 新手(甚至老手)写的代码。

问题出在哪?

  1. 耦合度爆炸: 你的 UI 组件在关心业务规则(> 500* 0.9)。如果老板改了需求:“哎呀,这次是满 1000 元打八折”,你不仅要改 JS 代码,还得改 UI 里的文案。如果需求是:“当总价超过 500 时,不仅打九折,还要发邮件给客服”,你现在的架构能做吗?不能。你只能硬塞进去。
  2. 不可测试: 你怎么测试 calculateTotal?你得先 useEffect 获取数据,再 render,再检查 DOM 里的 $ 符号?还是你写个单元测试,还得模拟整个 React 生命周期?太痛苦了。
  3. 状态混乱: discount 这个状态,它属于 UI 吗?属于业务吗?它好像既不是,也不是。它是个“副作用”的产物。

DDD 的核心思想很简单:业务逻辑界面展示剥离开。就像把“计算器”和“显示器”分开一样。计算器只管算,显示器只管显示。如果计算器坏了,显示器依然可以亮着;如果显示器坏了,计算器依然可以算出 100。

第二部分:洋葱架构——我们的架构蓝图

在 React 里实现 DDD,我们不一定要搞出个复杂的六边形架构图挂在墙上。我们用洋葱架构的变体就好。

想象一个洋葱:

  • 最外层(皮肤): React 组件、样式、路由。这是给用户看的,容易变。
  • 中间层(肉): 业务逻辑、服务、状态管理。这是核心,比较稳定。
  • 最内核(心): 实体、值对象、领域规则。这是灵魂,绝对不能动。

我们的目标就是:洋葱是圆的,但我们的代码文件是分层的。

第三部分:核心层——让数据活起来

在 DDD 里,数据不是简单的 interfaceclass,它们是有生命的。它们有自己的行为。

1. 实体

在数据库里,User 只是个记录。但在 DDD 里,User 是个实体。实体意味着它有一个唯一标识,并且它的行为是由它自己定义的。

比如,一个“库存”实体。它不应该只是有个 quantity 属性,它应该有个方法叫 decrease(amount)

不要写这样的代码:

// 坏代码
let stock = 10;
if (stock >= 5) {
  stock -= 5;
  console.log("剩余库存:", stock);
}

我们要写这样的代码:

// domain/entities/Inventory.ts
export class Inventory {
  private _quantity: number;

  constructor(quantity: number) {
    if (quantity < 0) {
      throw new Error("库存不能为负数!这是常识!");
    }
    this._quantity = quantity;
  }

  // 获取数量(只读,防止外部直接修改)
  get quantity(): number {
    return this._quantity;
  }

  // 核心业务行为:扣减库存
  // 这里包含了业务规则校验
  decrease(amount: number): void {
    if (amount <= 0) {
      throw new Error("扣减数量必须大于 0");
    }
    if (this._quantity < amount) {
      throw new Error("库存不足!别搞事情!");
    }
    this._quantity -= amount;
  }

  increase(amount: number): void {
    if (amount <= 0) throw new Error("增加数量必须大于 0");
    this._quantity += amount;
  }
}

看懂了吗?所有的验证逻辑都在方法内部。 这就是封装。UI 层根本不需要知道库存是否合法,它只需要调用 inventory.decrease(5),如果抛出异常,那就说明逻辑出错了。

2. 值对象

值对象是那些没有唯一标识,但描述了属性的对象。比如 Money(金额)。

为什么要在 JS 里搞个 Money 类?因为 10.99 + 10.99 等于 21.98 吗?在 JavaScript 的浮点数运算里,它可能等于 21.980000000000004。这会让财务系统崩溃。

// domain/value-objects/Money.ts
export class Money {
  constructor(private _amount: number) {
    if (_amount < 0) throw new Error("钱不能是负数");
  }

  add(other: Money): Money {
    return new Money(this._amount + other._amount);
  }

  multiply(factor: number): Money {
    return new Money(this._amount * factor);
  }

  // 格式化输出
  format(): string {
    return `$${this._amount.toFixed(2)}`;
  }
}

现在,你的业务逻辑里全是 Money 对象,而不是 number。这保证了财务计算的绝对准确性。

第四部分:应用层与领域层——服务与仓储

有了实体和值对象,我们需要一个地方来组织它们。这就是服务

1. 领域服务

领域服务用于处理那些不属于单个实体,但属于整个领域的逻辑。

比如“订单结算”。这个逻辑涉及到计算折扣、计算运费、检查库存、计算总价。这太复杂了,不能塞进 Order 类里,也不能塞进 Inventory 类里。

这时候,我们需要一个 OrderService

// domain/services/OrderService.ts
import { Inventory } from '../entities/Inventory';
import { Money } from '../value-objects/Money';

export class OrderService {
  constructor(
    private inventoryService: InventoryService, // 假设这是个仓储接口
  ) {}

  checkout(cartItems: CartItem[]): Result<Money> {
    // 1. 验证库存
    if (!this.inventoryService.checkAvailability(cartItems)) {
      return Result.fail("库存不足");
    }

    // 2. 计算总价
    let total = Money.from(0);
    cartItems.forEach(item => {
      total = total.add(item.price.multiply(item.quantity));
    });

    // 3. 应用折扣逻辑
    if (total._amount > 500) {
      total = total.multiply(0.9); // 九折
    }

    // 4. 扣减库存
    this.inventoryService.decrease(cartItems);

    return Result.ok(total);
  }
}

注意看,OrderService 里全是纯业务逻辑。它不关心 React 的 useState,不关心浏览器怎么渲染。它只关心:库存够不够?钱算对了吗?

2. 仓储接口

仓储是领域层和基础设施层的桥梁。它定义了“怎么存数据”,但具体怎么存(是存 Redis,还是存 MySQL,还是存 LocalStorage)由基础设施层决定。

// domain/repositories/IInventoryRepository.ts
export interface IInventoryRepository {
  findById(id: string): Promise<Inventory>;
  save(inventory: Inventory): Promise<void>;
  decrease(id: string, amount: number): Promise<void>;
}

为什么要接口? 为了解耦!如果有一天你老板说:“咱们别用 MySQL 了,改用 MongoDB 吧。” 只要你的 MongoDB 实现了这个接口,你的领域层代码(OrderService)一行都不用改!这就是 DDD 的魅力。

第五部分:UI 层——React 的正确打开方式

现在,我们有了最核心的领域层。UI 层应该做什么?

UI 层应该依赖领域层,但不包含领域层。

1. 适配器模式

我们要把领域层的逻辑“适配”到 React 上。

这里有一个关键点:不要在 UI 层写业务逻辑!

UI 层只负责:接收用户点击 -> 调用领域层服务 -> 获取结果 -> 更新 UI。

我们使用 React 的 Hooks 来实现这个连接。

// ui/hooks/useCheckout.ts
import { useState } from 'react';
import { OrderService } from '../../domain/services/OrderService';
import { IInventoryRepository } from '../../domain/repositories/IInventoryRepository';

// 定义一个 Hook,用来管理 Checkout 的状态
export const useCheckout = (inventoryRepo: IInventoryRepository) => {
  const [loading, setLoading] = useState(false);
  const [error, setError] = useState<string | null>(null);
  const [total, setTotal] = useState<number | null>(null);

  const service = new OrderService(inventoryRepo);

  const handleCheckout = async (cartItems: any[]) => {
    setLoading(true);
    setError(null);

    try {
      // 调用领域层服务
      const result = service.checkout(cartItems);

      if (result.isSuccess) {
        setTotal(result.data._amount);
      } else {
        setError(result.error);
      }
    } catch (err) {
      setError("未知错误");
    } finally {
      setLoading(false);
    }
  };

  return { handleCheckout, loading, error, total };
};

看,这个 Hook 里只有 loadingerrortotal。这些是 React 的状态。业务逻辑(checkout 的计算、库存检查)完全被隔离在 OrderService 里。

2. 组件实现

现在我们的 UI 组件就变得非常干净了:

// ui/components/CheckoutPage.tsx
import React from 'react';
import { useCheckout } from '../hooks/useCheckout';
import { IInventoryRepository } from '../../domain/repositories/IInventoryRepository';

// 模拟一个仓储实现(基础设施层)
class LocalStorageInventoryRepo implements IInventoryRepository {
  // ... 实现省略 ...
}

export const CheckoutPage: React.FC = () => {
  const cartItems = [
    { id: '1', price: 600, quantity: 1 }, // 总价 600,会触发折扣
    { id: '2', price: 100, quantity: 1 },
  ];

  // 注入依赖
  const repo = new LocalStorageInventoryRepo();
  const { handleCheckout, loading, error, total } = useCheckout(repo);

  return (
    <div className="checkout">
      <h1>结账</h1>
      <button onClick={() => handleCheckout(cartItems)} disabled={loading}>
        {loading ? '处理中...' : '提交订单'}
      </button>

      {error && <div className="error">{error}</div>}

      {total !== null && <div className="success">应付金额: ${total.toFixed(2)}</div>}
    </div>
  );
};

对比一下:

  • 以前的代码:你在 UI 组件里写 if (total > 500),修改 UI 还要改逻辑。
  • 现在的代码:UI 组件只是个传声筒。老板说“满 1000 打八折”,你只需要去改 OrderService 里的 multiply(0.8),整个系统自动生效,不需要动一行 JSX。

第六部分:深入探讨——如何处理异步与副作用

这是 React 开发者最容易晕的地方。在 DDD 里,我们怎么处理 API 请求?

答案:把 API 请求放在基础设施层,通过仓储接口暴露给领域层。

// infrastructure/repositories/ApiInventoryRepository.ts
import { IInventoryRepository } from '../../domain/repositories/IInventoryRepository';
import { Inventory } from '../../domain/entities/Inventory';

export class ApiInventoryRepository implements IInventoryRepository {
  async findById(id: string): Promise<Inventory> {
    // 这里是真正的网络请求
    const response = await fetch(`/api/inventory/${id}`);
    const data = await response.json();

    // 转换成领域实体
    return new Inventory(data.quantity);
  }

  async decrease(id: string, amount: number): Promise<void> {
    await fetch(`/api/inventory/${id}/decrease`, {
      method: 'POST',
      body: JSON.stringify({ amount })
    });
  }
}

注意到了吗?IInventoryRepository 接口里全是 async 方法。这意味着我们的领域层(OrderService)也必须变成 async

// domain/services/OrderService.ts (修改版)
export class OrderService {
  constructor(private inventoryRepo: IInventoryRepository) {}

  async checkout(cartItems: CartItem[]): Promise<Result<Money>> {
    // ...
    // 这里需要 await 仓储操作
    await this.inventoryRepo.decrease(cartItems[0].id, 5); 
    // ...
  }
}

然后在 UI 层的 Hook 里处理 await

const handleCheckout = async (cartItems: any[]) => {
  setLoading(true);
  try {
    const result = await service.checkout(cartItems); // 等待领域层完成
    // ...
  }
}

这就是清晰的分层:

  1. UI 层:处理用户交互,等待异步操作。
  2. 领域层:定义业务规则,调用仓储接口。
  3. 基础设施层:执行真实的网络请求,把数据变成领域实体。

第七部分:实战演练——重构一个“烂摊子”

假设我们有一个“用户注册”的模块。原来的代码是这样的:

// 烂代码
const RegisterForm = () => {
  const [email, setEmail] = useState('');
  const [password, setPassword] = useState('');
  const [confirmPassword, setConfirmPassword] = useState('');

  const handleSubmit = async (e) => {
    e.preventDefault();

    // 1. 简单校验
    if (password !== confirmPassword) {
      alert('密码不匹配');
      return;
    }
    if (password.length < 6) {
      alert('密码太短');
      return;
    }

    // 2. API 请求
    try {
      await axios.post('/api/register', { email, password });
      alert('注册成功');
    } catch (err) {
      alert('注册失败');
    }
  };

  return <form onSubmit={handleSubmit}>...</form>;
};

现在,我们用 DDD 重构它。

第一步:定义领域逻辑

// domain/entities/User.ts
export class User {
  constructor(public email: string, public password: string) {
    if (!email.includes('@')) throw new Error("邮箱格式不对");
  }
}

// domain/services/ValidationService.ts
export class ValidationService {
  static isPasswordStrong(password: string): boolean {
    return password.length >= 6;
  }

  static passwordsMatch(p1: string, p2: string): boolean {
    return p1 === p2;
  }
}

第二步:定义仓储

// domain/repositories/IUserRepository.ts
export interface IUserRepository {
  save(user: User): Promise<void>;
}

// infrastructure/repositories/ApiUserRepository.ts
export class ApiUserRepository implements IUserRepository {
  async save(user: User): Promise<void> {
    await axios.post('/api/register', {
      email: user.email,
      password: user.password // 实际上应该加密,这里略过
    });
  }
}

第三步:UI 层 Hook

// ui/hooks/useRegister.ts
import { useState } from 'react';
import { User } from '../../domain/entities/User';
import { ValidationService } from '../../domain/services/ValidationService';
import { IUserRepository } from '../../domain/repositories/IUserRepository';

export const useRegister = (repo: IUserRepository) => {
  const [email, setEmail] = useState('');
  const [password, setPassword] = useState('');
  const [confirmPassword, setConfirmPassword] = useState('');
  const [loading, setLoading] = useState(false);
  const [error, setError] = useState<string | null>(null);

  const handleSubmit = async (e: React.FormEvent) => {
    e.preventDefault();
    setLoading(true);
    setError(null);

    try {
      // 1. 领域逻辑校验
      if (!ValidationService.isPasswordStrong(password)) {
        throw new Error("密码太短,至少6位");
      }
      if (!ValidationService.passwordsMatch(password, confirmPassword)) {
        throw new Error("两次密码输入不一致");
      }

      // 2. 创建实体
      const user = new User(email, password);

      // 3. 调用仓储保存
      await repo.save(user);

      alert("注册成功!");

    } catch (err: any) {
      setError(err.message || "注册失败");
    } finally {
      setLoading(false);
    }
  };

  return { email, setEmail, password, setPassword, confirmPassword, setConfirmPassword, handleSubmit, loading, error };
};

看!

  1. UI 组件现在只需要处理表单输入、点击事件和显示错误。
  2. 业务规则(密码长度、密码匹配)被封装在 ValidationService 里。
  3. 数据模型(User)是强类型的。
  4. API 调用被隔离在 ApiUserRepository 里。

如果你的老板说:“我们要改成短信验证码注册,而不是邮箱密码”,你只需要:

  1. 新建一个 Phone 实体。
  2. 修改 ValidationService,增加手机号格式校验。
  3. 修改 ApiUserRepository,把 /api/register 改成 /api/register-phone
  4. UI 层的 useRegister Hook 只需要改一下 useState 的类型。

领域层代码,几乎不需要动!

第八部分:状态管理——不要让 Redux 变成你的噩梦

很多同学引入 DDD 的时候,都会纠结:“我到底该用 Redux 还是 Zustand 还是 Context?”

如果你的业务逻辑很重,而且需要跨组件共享,DDD 建议你使用Zustand或者Jotai这类轻量级状态管理库,或者干脆用React Context

但是!关键点来了:

在 DDD 架构下,你的状态应该是什么?

答案:状态 = 领域实体。

不要用 Redux 存一个 { name: "Alice", age: 25 }。存一个 User 实例。

// store/useUserStore.ts
import { create } from 'zustand';
import { User } from '../domain/entities/User';
import { IUserRepository } from '../domain/repositories/IUserRepository';

interface UserState {
  currentUser: User | null;
  login: (email: string, password: string, repo: IUserRepository) => Promise<void>;
}

export const useUserStore = create<UserState>((set) => ({
  currentUser: null,
  login: async (email, password, repo) => {
    // 这里可以调用领域服务进行登录验证
    const user = new User(email, password); // 模拟登录成功
    set({ currentUser: user });
  },
}));

这样,你的状态管理器里存的都是纯业务对象,而不是为了渲染 UI 而凑出来的数据。

第九部分:测试——DDD 的真正福利

这也是我最喜欢 DDD 的地方。

以前写测试,我得写个 render(<MyComponent />),然后用 screen.getByText 去找 DOM 元素。如果 UI 变了,测试就挂了。

现在,写单元测试,我只需要一个简单的 JS 文件:

// tests/domain/services/OrderService.test.ts
import { OrderService } from '../../domain/services/OrderService';
import { Inventory } from '../../domain/entities/Inventory';

describe('OrderService', () => {
  let service: OrderService;
  let mockRepo: any;

  beforeEach(() => {
    mockRepo = {
      checkAvailability: jest.fn().mockReturnValue(true),
      decrease: jest.fn(),
    };
    service = new OrderService(mockRepo);
  });

  it('should apply 10% discount when total > 500', () => {
    // 给仓储返回一个库存
    mockRepo.findById = jest.fn().mockReturnValue(new Inventory(100));

    const result = service.checkout([{ id: '1', price: 600, quantity: 1 }]);

    expect(result.isSuccess).toBe(true);
    expect(result.data._amount).toBe(540); // 600 * 0.9
  });

  it('should fail if stock is insufficient', () => {
    mockRepo.findById = jest.fn().mockReturnValue(new Inventory(0));
    mockRepo.checkAvailability = jest.fn().mockReturnValue(false);

    const result = service.checkout([{ id: '1', price: 100, quantity: 1 }]);

    expect(result.isSuccess).toBe(false);
    expect(result.error).toBe("库存不足");
  });
});

没有 React,没有 DOM,没有 <div />。只有纯粹的逻辑判断。这就是写测试的乐趣。

第十部分:如何起步——别贪多

我知道,看到这里你可能会想:“哇,要建这么多文件夹,写这么多类,我的项目会不会变得很重?”

答案是:是的,你的项目会变重。

但是,这种“重”是有价值的重

起步建议:

  1. 从一个小模块开始: 别试图重构整个 App。找一个独立的模块,比如“购物车”或者“用户设置”。
  2. 实体先行: 先把 UserProduct 这些类写出来,定义好它们的行为。
  3. 提取服务: 把那些 useEffect 里的逻辑提取出来,放到 Service 里。
  4. 慢慢迁移: UI 层慢慢去调用这些 Service。

记住: React 是个 UI 框架,不是业务框架。你的核心价值在于你解决业务问题的能力,而不是你写出多炫酷的 CSS 动画。

结语

各位同学,今天我们聊了很多。我们抛弃了“把所有东西都塞进组件里”的坏习惯,拥抱了洋葱架构。

我们看到了如何用 TypeScript 类来封装业务规则,如何用仓储模式解耦数据访问,如何让 UI 层变得纯粹而简单。

DDD 不是一种魔法,它是一种纪律。

它要求你:

  1. 诚实: 承认你的业务逻辑很复杂,不能只靠 UI 层来扛。
  2. 分离: 把业务逻辑从 UI 逻辑里剥离出来。
  3. 封装: 让数据对象自己管理自己的行为。

当你下次看到那个长达 800 行的 App.tsx 时,别再硬着头皮改了。深吸一口气,打开你的编辑器,新建一个 domain 文件夹。

把那个该死的逻辑,搬出去。让它们在阳光下奔跑,而不是躲在 UI 的阴影里瑟瑟发抖。

好了,今天的讲座就到这里。下课!

(记住,写代码的时候,别忘了保存。还有,把你的 any 类型都改成 T,或者更好的,stringnumber。这是对你自己的尊重。)

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注