(5) 基于领域分析设计的架构规范 - 充血模型之Service

Entity与Service,相爱相杀

好,接上一篇。

既然采用order.cancel()这种模式,那么一个新的问题来了:

所有的命令操作都要变成这样子吗?那曾经巨大的OrderService的代码,岂不是只是单纯挪了一个位置,放在Order里面了,除了上面所谓的可读性的优势,那还有什么用?

并不是,只是一部分放在实体类,其余的命令操作,依旧会采用一种Service来做。
所以,我们必然需要一个可以清晰量化的规范,来确定这些行为该放在哪里。
好的,那就来详细说明一下规范:

如果一个命令操作,只修改了一个聚合内部的相关数据,那么,就归属给这个聚合
比如,订单取消这个行为,需要做的事情有:

  1. 订单状态标记为取消
  2. 订单变更记录插入一条,“订单取消”

根据我们之前的图可以知道,这些修改操作,都在这个订单聚合内,很自然的归属给order

订单聚合

注意,我们反复强调了这里是“修改操作”,也就是说,如果需要我们在此操作期间,查询其他聚合的信息,只要不做修改,那就是允许的!就像下面这样:

@Entity
public class Order{

  private OrderStatus status;
  private String customerName;
  private User orderCreator;   //假定这里是下单用户,省略了many-to-one的配置
  //...
  
  public void cancel(){
  
      //修改操作1:变更订单自身状态
      status = OrderStatus.CANCELLED;
      
      //查询用户信息,要记录到订单变更日志中,
      //这里如果是Hibernate,会直接触发sql查找,如果换成其他如mybatis,则用对应的repository操作即可,总之,是一个纯查询
      String userName = orderCreator.getUserName();
    
      //修改操作2:增加一条订单状态变更信息,具体实现省略
      createOrderTrack(OrderStatus.CANCELLED,userName);
  }
}

然后,就要从另外一个角度来说了
如果一个命令操作,并且要求是一个完整的事务,修改了多个聚合的数据,那么,需要为这个行为建立一个 Service
而这个Service,不会是一个{领域名称}+Service,而是一个{具体动作}+Service,比如OrderPayService,订单支付,假定有如下动作:

  1. 订单状态改为支付中
  2. 商品库存对应扣减
  3. 用户若使用了优惠券,则优惠券标记为使用中
    这几个操作,是要在一个完整的事务中的,所以我们写在一个Service中
@Transactional
public class OrderPayService{                            //-----------(1)

    @Autowired OrderRepository orderRepository;          //-----------(2)
    @Autowired CouponRepository couponRepository;

    public String execute(Long orderId,Long couponId){   

        //暂不考虑前置状态检查

        //订单属性变更
        Order order = orderRepository.getById(orderId);
        order.setStatus(OrderStatus.PAYING);

        //商品库存扣减,按之前的假定,一个订单只对应一个商品
        Prodect product = order.getProduct();
        product.minusStock(order.getQuantity);           //-----------(3)

        //变更优惠券状态
        Coupon usingCoupon = couponRepository.getById(couponId);
        usingCoupon.setStatus(CouponStatus.USING);

        //去交易中心获取支付unikey
        CreatePayResponse payResponse = payCenterApi.createPay(...各种参数...);
        return payResponse.getUnikey();
  }
}

//这时,上层入口(如Controller)就是这样调用了
orderPayService.execute(orderId,couponId);

好,老规矩,深入探讨一下:

  1. 类被命名为订单支付服务,也就是{一个动作}+服务,代码的清晰性上来说不言而喻,但也意味着一个操作就要有一个service,其实这是非常符合单一指责原则的,但是肯定会有不少同学觉得这样做是容易产生过多的类,过度设计了。确实,会有这种情况,但我依旧推崇这样做,或者说,如果一定要一个service里多个行为,那至少表示这个行为是相关的比如都是订单,但是PC端下单APP下单等等,可以放在一起,这样职责不泛滥,也便于代码复用
  2. service,我们可以给与其足够的权限,只要它需要,它可以无所顾忌地获取所有的上下文组件,不管是jdbc组件,还是外部rpc组件,都是可以的。因为给它的定义,本来就是多聚合的事务处理类,所以,只要它能保证事务的安全性,保证业务的完整,这一切都是没问题的(这里暂时不讨论分布式事务问题,那是另外要一个议题)
  3. 库存扣减,我们这里采用了product.minusStock(quantity),而不是直接对product进行属性修改。当然,直接进行属性修改也是可行的,但是为何这里却封装成了一个方法呢?很可能的原因是,最早的时候,是直接改属性的,但后来有很多地方都要扣减库存,所以,代码重构了,然后minusStock应运而生。关于,重构,我们后面还会提到。

关于Entity的Set方法

如果我们使用很多ORM框架,由于框架的实现策略的缘故,实体类是需要把所有的Get和Set方法都要开放的,而且上面大家也看到当我们用OrderPayService的时候,也直接使用的对象的setXXX方法,所以Set自然更加需要开放了。

但对于set,本文这套规范,极力倡导一个原则:在进行业务开发时,Set能调用的地方只有1个,那是就在service中! 其余的任何场景,任何地方,都不允许(或者没必要)调用set方法,尤其是下面这种场景:

//在一个上层,比如Controller中
@GetMapping("/coupon/disable/{id}")      //失效某张优惠券,偷懒就不用Post了
public ActionResponse disableCoupon(@PathVariable("id") Long id){
      Coupon coupon = couponRepository.getById(id);
    
      //错误,禁止!!!
      coupon.setStatus(CouponStatus.DISABLED);
    
      //正确的应该是
      coupon.disable();
}

public class Coupon{
    
  private CouponStatus status;
  
  public void disable(){
      status = CouponStatus.DISABLED;
  }
}

一定会有同学马上提出疑问

才一行代码,为什么不能直接用set?强迫症吗?

不否认,这个规范,的确有点强迫症,但是真的是有好处的。

领域设计的思想里,严格意义上来说,Get和Set都是不能随便暴露的,尤其是Set,是在修改这个系统,是有一定风险与危害的,那么,任何一个set,都一定是有原因的,一定是要归属到一个具体的业务命令操作中的。

其实,我在思考这套规范期间,一度将set方法直接设置成本包可见的级别,希望通过Java编译报错来杜绝这种情况,但是这样又和上面提到的service的模式出现了冲突,最终只能作罢。

工厂

到此为止,我们可以认可,所有的命令操作,都将会归类到Entity或者Service中

但有一个特例,这里有必要提出来单独说一下:一个实体的创建,也就是增删改查中的 增 的操作

因为删,改的操作,都是先找到一个实体,然后进行操作。但创建却不同,因为在执行创建操作之前,这个实体都是不存在的,你怎么找?就更加不可能有类似 order.create(params)这种代码出现了,order尚且不存于世呢!
所以,这里,自然的想到通过创建OrderCreateService来处理,但考虑到新增的特殊性,建议直接用工厂模式来做,即OrderFactory

对于OrderFactory,它的责任并非简简单单new Order()然后一堆setXX后完事。详细说明如下:

  1. 负责创建聚合根对象,比如订单聚合中的Order,往往创建会有诸多不同的场景,比如创建一个空对象,或者创建有很多默认组件的对象等等,这个就根据业务场景来了。总之返回值一定是一个新创建的实体类。
  2. 负责创建聚合中其他对象,但是这种场景仔细想来,并不会太多。因为聚合中其他“附属实体”的创建往往会以聚合根实体的某一个命令操作相关。比如订单变更记录OrderTrack的创建往往是伴随Order的各种各样的操作,比如订单创建,支付,发货,取消等等,而大部分时候不需要单独出现诸如OrderFactory.createOrderTrack的情况。
  3. Factory中原则上是允许触发对其他领域聚合的数据变更的,因为它是一个特殊的领域Service。但从一般的业务场景来说,这种情况并不多见,因为创建后立即变动某个其他领域的数据,往往会直接在应用层加入代码,或者通过事件来处理(事件后文会介绍)。

事件

领域事件,在最早的《领域驱动设计》一书中,并未提及。在之后的相关书籍,诸如《实现领域驱动设计》中,有将其作为一等公民的身份进行详细讲解。很惭愧,这一块我一直没有GET到其精髓,所以我只是结合更广义上的事件,来做了一些分析,如有不合适的地方,也欢迎各位拍砖。

说到事件,大家一定能联想到事件的广播,事件的处理。没错,事件是一个非常好的解耦和工具,也是一个非常舒服的“梳理工具”,因为在讨论需求的时候,经常能够看到非常“顺理成章”的事件场景描述:

当用户创建了一个订单后,要同时生成一个订单变更记录 ------ (1)
当用户的订单支付成功后,要同时为这个用户生成一个XX奖励券,并且用户活跃积分+10   ------ (2)

这一些,都太符合“事件”了,这些有些是在项目第一版的时候就清楚了,有些则随着版本迭代不断加入。
但是回过头的仔细想想,如果遇到这种需求场景的时候,大家在实际的开发中,都真的用了事件吗?不论是单机应用还是分布式应用,我们在不加入事件机制的前提下,上面这些功能通过直接“调接口”都是完全能满足的。

没错,所以对于事件模式,我们可以倡导一个原则:所有的事件,从重构中得来
从重构中得来,意味着,我们没必要在需求一开始就大量采用事件的做法,即使需求描述中有“当/如果...就...”,因为绝大部分时候,我们往往无法得知之后的产品的发展方向是什么,过早的事件设计(尤其是分布式系统)会给代码的阅读流畅性,事务管理等等带来更大难度。

事件的优势在于“一处广播,多处接收”,所以,当“接收方”越来越多的时候,也是事件机制的优势能体先出来地时候。所以,我认为最佳的实践方式,或者更容易推广的实践方式,还是跟着版本迭代来不断优化代码,在逐渐清晰地产品发展方向和扩展方向上,将原有的“直接调用”转变成“事件处理”。一般来说,当“接收方”出现2-3个的时候,可以开始考虑转变成事件机制了,比如上面的(2)

当然,这里并不否认在第一时间就加入事件机制的做法,只是建议如果确定要在一开始就这样做,希望这种做法的开发负责人务必对业务的扩展方向有足够清楚的认识与了解。(比如一家公司在原有系统上开发新的升级版系统,这个时候,可以在第一时间就做好设计优化,因为有很好的业务背景基础)

至于具体的代码实现方式,在单机应用中,Spring有很好的事件机制,而且能够支持事务的完整性。而分布式系统中,更多的接用消息中间件来实现。

下一篇 关于重构

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351