当前位置: 首页 > news >正文

让代码变美的第一天 - 观察者模式

文章目录

  • 丑陋的模样
  • 变美步骤
    • 第一步 - 基本预期
    • 第二步 - 核心逻辑梳理
    • 第三步 - 重构
      • 重构1 - 消息定义
      • 重构2 - 消息订阅
      • 重构3 - 消息发布
    • 高级用法
      • 按顺序订阅
      • 异步订阅
      • 多消息订阅

丑陋的模样

当我们开发一个功能,代码可能如下:

private void test() {func1();func2();func3();
}
  • 目前看起来还是美美哒,结构清晰,层次分明
    新需求来的时候,慢慢的变成了下面这样:
private void test() {func1();try {func2();} catch (Exception e) {}func3();func4();// func5if (1 == 1) {for (int i = 0; i < 5; i++) {// todo1// todo2}}
}
  • 多人开发的话,情况更严重。当项目大了里面的逻辑很少有人能全部讲清楚,老代码没人敢动(劝你也不要动,一年到头辛辛苦苦,背个故障从头来过)
  • 代码会越来越乱,在一个方法里面堆砌的代码会越来越多,如果有洁癖一点的可能还会把新代码封装成func4,如果偷懒可能就变成了func5()下面的代码了,屎山代码可能就是这样出来的吧

变美步骤

第一步 - 基本预期

思考:这个功能,这个方法改动到底是不是很频繁。有个预期就行,如果没有预期,那就看实际改动次数。

  • 如果不频繁,建议就这样吧,只要结构清晰,方便review代码就行。不鼓励过度设计,毕竟开发时间都是有限的。
  • 如果很频繁,就看第二步

第二步 - 核心逻辑梳理

还是上面那个代码,需要思考到底哪部分是核心,假设func1、func2、func3是库存校验、下单、修改库存。func4、func5是销量统计、发货
那么很容易判断出,func123才是核心逻辑,是下单的必要步骤。func45就是下单完之后的一些后续处理,甚至可能都不需要同步处理,比如发货,和下单并没有强绑定。
很容易就判断出来,这可能是一个发布订阅的逻辑,可以用观察者模式来重构,学了这么多年的设计模式,这不就能用上了么。

第三步 - 重构

  • 如果未来计划很明确,项目越做越大,肯定要拆分成多个微服务,组成一个分布式服务,并且已经有消息中间件了(Kafka、RocketMQ等),那么就直接上吧,用发布订阅来异步、解耦、削峰。
  • 如果还没有那么明确的目标,也不想引入大型中间件,那么重构一下代码还是很不错的。

重构1 - 消息定义

只讨论使用spring项目

public class OrderEvent {private long id;private String desc;
}
  • 需要向下传递的参数信息

重构2 - 消息订阅

@Service
public class OrderEventListenerService {@EventListenerpublic void handleForOrderSaveEvent(OrderEvent event) {log.info("收到接单消息:{}", event);}
}
  • 这里订阅的就是func4/5

重构3 - 消息发布

@Service
public class OrderEventPublishService {@Autowiredprivate ApplicationEventPublisher eventPublisher;public void order() {OrderEvent event = new OrderEvent(1, "接单");log.info("发送接单消息:{}", event);eventPublisher.publishEvent(event);log.info("消息发送完毕:{}", event);}
}
  • 这里就是func1/2/3
    三步就搞定了重构,并且下次修改,只需要新增一个订阅方法。

高级用法

按顺序订阅

func4、func5是需要有先后顺序的,那么只需要加上@Order(num),按到num从小到大的顺序先后处理

@Service
public class OrderEventListenerService {@Order(2)@EventListenerpublic void handleForOrderSaveEvent1(OrderEvent event) {log.info("2同步收到接单消息:{}", event);}@Order(1)@EventListenerpublic void handleForOrderSaveEvent2(OrderEvent event) {log.info("1同步收到接单消息:{}", event);}
}

异步订阅

如果func5是发货,完全不依赖下单,订阅方就可以异步处理,使用@Async就行了。

@Service
public class OrderEventListenerService {@Async@EventListenerpublic void asyncHandleForOrderSaveEvent(OrderEvent event) {log.info("异步收到接单消息:{}", event);}
}

注意:使用的是默认线程池,最好改成自定义的线程池

多消息订阅

如果func5是发货,除了C端用户下单以外,B端用户也能下单,但是B端下单代码完全是另一份,那么就需要有B端的下单消息,func5统一处理发货流程就行了

@Service
public class OrderEventListenerService {@Async@EventListener(classes = {OrderAEvent.class, OrderBEvent.class})public void asyncHandleForOrderSaveEvent(Object event) {log.info("异步收到接单消息:{}", event);}
}
http://www.lryc.cn/news/224444.html

相关文章:

  • 微服务-网关设计
  • WxJava使用lettuce的redis实现access_token的共享
  • 干货:如何运作一个全新品牌?
  • TCP/IP卷一详解第二章Internet地址结构概要
  • 小程序 打开方式 页面效果 表单页面 点击跳到详情页 图标 获取后台数据 进行页面渲染
  • 一个“Hello, World”Flask应用程序
  • 计算机丢失mfc100.dll如何恢复,详细解析mfc100.dll文件丢失解决方法
  • 分享一本让你真正理解深度学习的书
  • Apache APISIX Dashboard 未经认证访问导致 RCE(CVE-2021-45232)漏洞复现
  • Git 安全警告修复手册:解决 `fatal: detected dubious ownership in repository at ` 问题 ️
  • 【MySQL事务篇】多版本并发控制(MVCC)
  • 拆分代码 + 动态加载 + 预加载,减少首屏资源,提升首屏性能及应用体验
  • 在 Vue3 中使用 mitt 进行组件通信
  • SQLite 3.44.0 发布!
  • 本地生活新赛道-视频号团购怎么做?
  • 输入一个url后,会发生什么事?
  • R语言和jsonlite库编写代码示例
  • 容联七陌携手岚时科技,解决医美机构回访3大痛点
  • 自动计算零售数据分析指标?BI软件表示可行
  • Qt读取xml文件并把内容显示到QTableview上
  • xv6-x86在ubuntu14.04 i386下正常编译、调试,在ubuntu23.04下编译各种报错--google镜像
  • 关于unity中 编辑器相关逻辑的记录
  • linux安装配置MongoDB并设置开机启动
  • 1366 - Incorrect string value: ‘\xE5\xB9\xBF\xE5\x85\xB0...‘ for column编码错误
  • K8S篇之谈谈kubelet的上报机制
  • 混沌系统在图像加密中的应用(小波混沌神经网络)
  • Node.js中的文件系统(file system)模块
  • react组件间通信之context
  • 京东数据分析:2023年10月京东洗衣机行业品牌销售排行榜
  • QQ恢复聊天记录,就用这3个方法!