DDD中domain落地的思考

定时任务扫描订单数据

任务描述:

设计为主任务和子任务,主任务只处理指定时间片,子任务会拆分主任务的时间片。其目的为拆分后的子任务可以并行执行,不会干扰,提供吞吐量。

主任务状态:新建、执行中、执行完毕
多次触发时,没有主任务则新建,执行中不处理,当前任务执行完毕,则需新建下一个时间段的主任务。

主任务初始化:
触发 job,主任务领域通过【加载最近执行的任务】或【新建主任务】方式,初始化主任务信息。

//获取最近同步的主任务  
OrderSyncMainTask lastSyncMainTask = MainTaskFactory.getOrInitLastMainTask();
public static OrderSyncMainTask getOrInitLastMainTask() {  
    // 任务查询并启动,暂时未考虑连续触发的情况(并发)  
    OrderSyncMainTask orderSyncMainTask = omsOrderSyncTaskRepo.getLastMainTask();  
    if (orderSyncMainTask == null) {  
        orderSyncMainTask = initMainTask(null);  
    }  
    return orderSyncMainTask;  
}

问题描述:
这里【主任务领域】初始化的方式,采用工厂+静态方法,对于外部传入的入参,是很容易加工生成目标类的,但如果需要再次读取 db 库的数据,工厂类就需要引入(注入)相关 service、repository。

@Slf4j  
@Component  
public class MainTaskFactory {
    static OmsOrderSyncTaskRepo omsOrderSyncTaskRepo;
    //从spring容器找到要用的bean
    @Autowired  
    public void setOmsOrderSyncTaskRepo(OmsOrderSyncTaskRepo omsOrderSyncTaskRepo) {  
        MainTaskFactory.omsOrderSyncTaskRepo = omsOrderSyncTaskRepo;  
    }
    //...
}

现在出现一个我们不熟悉的依赖方式,静态工厂的静态属性,需要引入 spring 的 bean 实例。

实际上,让 MainTaskFactory 做了初始化,从持久化数据库读数据,就必然依赖 spring 中的 bean:


image.png

整理下 MainTaskFactory 的工作职责:

  • 职责 1: 为了通过静态方法使用其它 bean,将自己作为 bean 组件,并通过 static set 方式引入要用的 bean。
  • 职责 2:获取、初始化 MainTask。
    • create MainTask,实际上 new MainTask
  • 职责 3:持久化主任务,为此需要职责 1。

总结为:factory 做了查询、构造、保存工作,这里分工混乱。

谁来读取持久化的数据(分工);
如何读取持久化的数据(实现方式);

谁来保存 domain 信息;

repo.save(domain)

domain 转换出 dto1,dto2,dto3
=======以下为同一事务=======
dto1Repo.save(dto1);
dto2Repo.save(dto2);
dto3Repo.save(dto3);

写到这里,发现写和读应该属于同一逻辑:

//repo 加载 domain 的方法:
public domain load(cmd.getId);

按以上想法,重新设计:

repo 层,从持久化服务中,加载需要的 dto,交给 factory 构造 domain。

public SyncOmsOrderTaskDomain loadLast() {  
MainTaskDTO mainTaskDTO = null;//omsOrderSyncTaskRepo.getLastMainTask();  
List<SubTaskDTO> subTaskDTOList = null;//omsOrderSyncTaskRepo.getSubTaskList(mainTaskDTO.getId());  
List<SubTaskRecordDTO> subTaskRecordDTOList = new ArrayList<>();//获取子任务执行记录,存在多个记录,取最后一次,或者设计上,保证每个子任务只有一个有效的记录  
return MainTaskFactory.create(mainTaskDTO, subTaskDTOList, subTaskRecordDTOList);  
}

factory 构造 domain 对象。

    public static SyncOmsOrderTaskDomain create(MainTaskDTO mainTaskDTO, List<SubTaskDTO> subTaskDTOList, List<SubTaskRecordDTO> subTaskRecordDTOList) {
        String taskId = UUIDHexGenerator.generate();
        //对查询的值做一些基本校验
        return new SyncOmsOrderTaskDomain(mainTaskDTO, subTaskDTOList, subTaskRecordDTOList);
    }

domainService 中,操作 domain 的各种方法,包括必要时,会通过入参、配置等信息,通过 factory 构造 domain。

@Service
public class MainTaskDomainService {
    @Resource
    MainTaskRepo mainTaskRepo;
    @Resource
    MainTaskConfig mainTaskConfig;
    @Resource
    SubTaskConfig subTaskConfig;
    @Resource
    ConcurrentReadOrderService concurrentReadOrderService;
    //触发任务调度
    public String triggerTask() {
        SyncOmsOrderTaskDomain syncOmsOrderTaskDomain = mainTaskRepo.loadLast();
        if (syncOmsOrderTaskDomain.ifNotExsit()) {
            syncOmsOrderTaskDomain.initTaskByConfig(mainTaskConfig);
            syncOmsOrderTaskDomain.initSubTaskByConfig(subTaskConfig);
            mainTaskRepo.save(syncOmsOrderTaskDomain);
            //细节1:实时更新子任务发送进度,如何实现?
            // 1。子任务查询并发送成功后,可使用内部通知机制,异步通知、更新。
            // 2。子任务查询发送、直接调用repo更新(有点耦合)
            concurrentReadOrderService.executeRead(syncOmsOrderTaskDomain);
            return "任务首次运行";
        } else {
            if (syncOmsOrderTaskDomain.isRunning()) {
                return "执行中";
            }
            if (syncOmsOrderTaskDomain.needReRun()){
                syncOmsOrderTaskDomain.updateReRunTask();
                mainTaskRepo.updateReRunTask(syncOmsOrderTaskDomain);
                concurrentReadOrderService.executeRead(syncOmsOrderTaskDomain);
                return "重新运行";
            }
            if (syncOmsOrderTaskDomain.isFinished()) {
                syncOmsOrderTaskDomain.createNextTask();
                mainTaskRepo.save(syncOmsOrderTaskDomain);
                concurrentReadOrderService.executeRead(syncOmsOrderTaskDomain);
                return "运行下一周期";
            }
        }
        return "未知状态";
    }
}

更新子任务发送状态:

执行子任务时,会组装成 runable,交给线程池执行查询,每次查询到数据,同时发送内部事件,更新子任务的发送数据数以及更新是否发送完毕状态。
发送消息结构:

class SubTaskSendCommand {
    String taskId;
    String subTaskId;
    int sendCount;
    boolean sendFinished;
}

因为发送的状态数据仅需保存,所以通过 factory 直接构造 domain,通过 repo save。

SyncOrderTaskDomain domain = MainTaskFactory.generate(subTaskSendCommand);

repo.updateSendInfo(domain);

更新子任务消费状态:
扫描的数据被消费处理成功后,需要更新任务记录状态,处理流程和上步类似:

class SubTaskConsumerCommand {
    String taskId;
    String subTaskId;
    int receivedCount;
}
SyncOrderTaskDomain domain = MainTaskFactory.generate(subTaskConsumerCommand);

repo.updateReceiveInfo(domain);

整理流程都收敛在 domainService 层,factory 主要职责是构造 domain 对象,repo 负责加载、保存、更新 domain 信息。

domain 处理各类 command 指定,包括任务触发、更新任务发送记录,更新任务消费记录。

关于 domain 内部模型,如何管理主任务、子任务、任务记录,后序更新。

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

推荐阅读更多精彩内容