浅析:通过自定义DSL实现一个序列号生成器
原始文章首发于个人博客
一、什么是DSL
DSL是(Domain Specified Language)的简称,中文含义为:领域专用语言。
设计者通过特定的语义,描述一些在特定的应用场景中出现的东西。
二、为什么要使用DSL
设计并使用DSL的优势在于:在解决特定问题时,有更简洁、更强大的语义表达能力。可以使用更少的代码(或配置)来描述问题,开发效率高。
我们在日常工作中,往往会在大量重复性的工作上浪费大量时间。针对这种共性很强的工作,我们设计一个具有通用性和简洁性的描述性语言,可以减轻我们的工作负担,也可以使项目代码更简洁易读。
三、分析业务需求
在公司内部系统开发中,有一个这样的需求:
需要设计一个序列号生成器,它能放在不同模块代码中,根据不同模块的序列号生成规则生成序列号。
通过分析生成规则,发现序列号生成器需要包含的功能有:
1. 在不同业务模块中生成的模块CODE不一样
2. 需要包含当前日期。但生成的日期格式每个模块可能会不一样。(有的是yyyy-MM-dd,有的是yyyyMMdd等)
3. 需要生成流水号。流水号各个模块需要生成的位数可能不一样,有的需要定长4位,有的需要定长3位。位数不足的有的需要左补0(或者左补其他符号)
四、DSL设计
通过上述的需求分析,我们不难看出:
- 各个业务模块的序列号生成策略有一定的共性:比如都可能包含:日期、流水号等
- 序列号生成策略中的变量仅有:模块编码、日期、流水号这三项,且业务模块对于流水号的需求仅定长不同
综上所述,便有了2种解决思路:
- 设计一个序列号生成工厂组件,提供:模块编码、日期、流水号等生成策略选项,由开发人员在实际开发过程中自己调用相关方法,按需使用。
- 考虑到每个模块都对应一个唯一的模块编码(模块编码要有,但是可以不作为生成序列号的一部分被显示),因此可以将业务模块和对应的模块编码相绑定。然后再利用注解,在对应的模块编码上指定相应的生成策略。
第一种“手动指定策略”的思路我们先不详细说。这里来详细说说第二种思路。
- 先将各个业务模块对应的模块编码梳理出来,整理成一个配置类
- 在配置类中的模块编码上,使用注解指定该模块对应的序列号生成规则
- 打算使用类似于EL表达式中取值的方式(即:类似于:xxx的方式)(注:后续代码中为了方便,直接用了的方式来取值,就没有写{xxx}的方式)(注:后续代码中为了方便,直接用了{}的方式来取值,就没有写xxx的方式)(注:后续代码中为了方便,直接用了的方式来取值,就没有写了)来实时计算并填充数据
- 将计算出来的序列号返回
因为在业务需求中只有4个变量,所以我们首先得给那4个变量起个名字:
例如:
- 模块编码可以使用: ${module_code}来表示
- 日期可以使用:${date}来表示
等等…
五、详细代码设计
- 首先准备一个配置类,将梳理好的模块编码都放进去
- 使用自定义注解指定序列号生成策略
如上图所示,电脑前有新的同学肯定已经发现了:里面还内嵌了一个@Appender注解。啥是@Appender注解?
@Appender注解定义长这样:
这是因为考虑到后期的代码扩展性,我将流水号生成的策略(占几位,补什么符号等)抽离出来,使用@Appender注解可以起到灵活配置的作用。
- 写一个脚本解析器,用于解析注解中的序列号生成策略
首先读取注解内容
计算或获取实际内容,然后替换掉生成策略脚本中对应的变量占位符
(注:为了精细化控制日期显示格式,所以在代码中用了多种日期格式变量)
字符替换方法详情:
六、总结
上面只是简单地谈一谈一种基于自定义DSL,结合注解形式来写一个序列号生成器地思路。还有很多细枝末节的东西没有说到,例如:
- 如何保证每天生成的流水号从1开始顺序累加? --redis或其他分布式锁机制
- 如何用最短代码将多个模块共用同一套生成策略?
- 如何同时集成“自动生成”和“手动生成”两种方式?
如果展开说那又可以水一篇博文了,这篇文章主要说一下思路,也算是给自己做一个经验累积~
浅析:通过自定义DSL实现一个序列号生成器相关推荐
- 基于美团Leaf-Segment的双buffer方案实现序列号生成器
业务背景 有时项目中对于流水号有一些特殊的需求.比如,和业务A有关数据,我们在落库时想要给每条数据添加一个流水号字段,用于作为全局唯一标识.流水号格式规则如下,如:BTA(业务A代号)+年月日(202 ...
- 加密解密软件VMProtect教程(八)许可制度之序列号生成器
VMProtect是新一代软件保护实用程序.VMProtect支持德尔菲.Borland C Builder.Visual C/C++.Visual Basic(本机).Virtual Pascal和 ...
- 微信序列号生成器架构设计及演变
一.摘要 微信在立项之初,就已确立了利用数据版本号实现终端与后台的数据增量同步机制,确保发消息时消息可靠送达对方手机,避免了大量潜在的家庭纠纷.时至今日,微信已经走过第五个年头,这套同步机制仍然在消息 ...
- 用一个程序生成另一个程序_还有另一个报告生成器?
用一个程序生成另一个程序 如果您具有业务应用程序开发的经验,那么很可能会遇到要求该应用程序具有灵活的报告机制的需求. 我工作的公司主要专注于开发业务解决方案,而报告是必不可少的,实际上,它必须包含我们 ...
- 还有另一个报告生成器?
如果您具有业务应用程序开发的经验,那么很可能会遇到要求该应用程序具有灵活的报告机制的需求. 我工作的公司主要专注于开发业务解决方案,而报告是必不可少的,的确,它必须包含我们开发的所有企业系统的方面. ...
- 有趣的深度学习——使用TensorFlow 2.0 + RNN 实现一个古体诗生成器
一.前言 很早之前,我曾经写过一个古体诗生成器(详情可以戳TensorFlow练手项目二:基于循环神经网络(RNN)的古诗生成器),那个时候用的还是Python 2.7和TensorFlow 1.4. ...
- Archive of Our Own_开箱用out-of-box Redis序列号生成器 不再写任何代码 你值得拥有
把简单的东西"傻瓜化"是软件开发追求的目标之一.请看下图: 左边是在 application.yml 里配置了3个生成器,右边可以直接注入到代码中使用,注意,不用写任何代码.这酸爽 ...
- c语言中定义密码为英文字母,请设计 一个密码生成器,要求随机生成4组10位密码(C语言)...
请设计 一个密码生成器,要求随机生成4组10位密码(密码只能由字母和数字组成),每一组必须包含至少一个大写字母,每组密码不能相同,输出生成的密码. #include #include #include ...
- 基于单例模式的多键值序列号生成器实现(支持缓存)
之前在BlogJava上发表过这篇文章,那时没怎么做整理.想不到已经有博友把它弄到ITEye来了(虽然写得水平差,但还是希望那位博友能注明是转载的).这次稍作了整理,自己也加深下印象. 使用场景:用于 ...
- 万亿级调用系统:微信序列号生成器架构设计及演变
"每天万亿级调用的重量级系统,每次申请序列号平时调用耗时1ms,99.9%的调用耗时小于3ms,服务部署于数百台4核CPU服务器上!" 老司机介绍: 曾钦松,微信高级工程师,目前负 ...
最新文章
- PHP计划任务:如何使用Linux的Crontab执行PHP脚本(转)
- LeetCode实战:螺旋矩阵
- jeecg如何控制用户权限_如何为用户定制个性化控制台?
- Oracle数据库中序列(SEQUENCE)的用法详解
- 个人推荐的SIP software Phone
- ssh本地端口转发,远程端口转发,隧道(这个解释不饶)
- 安兔兔2019年1月安卓手机排行榜发布:小米无愧性价比之王
- 思科CCNA考试流程-ielab
- 安卓TV开发(前言)— AndroidTV模拟器初识与搭建
- 图片放大后怎么把图片变清晰?
- struts2与jasperreport整合其余问题
- PS技巧一-----镜头光晕
- 【Touch】触摸事件onTouch
- root后手机银行用不了,root之后手机银行不能用
- 2020软件开发趋势预测(学习总结)
- docker privileged
- H5项目(基于vue框架)常见问题及注意事项
- C#将文本用语音朗读
- ContextCapture User Guide V4.4.11 Home(Smart3D 帮助文档 目录)
- 扑克牌游戏java代码_java实现简易扑克牌游戏
热门文章
- 2021南京大学软件学院考研经验
- Linux线程-概念和控制
- vcpkg快速入门手册
- Python3学习笔记_F(垃圾回收)
- grubbs准则 matlab_MATLAB-格拉布斯准则(MATLAB-Grubbs criterion)_0
- Brother DCP-1608 Printer共享打印机防坑指南
- java测试类调用方法_java – 在单元测试中调用其他类方法
- PPT中的文字怎么填充为图片?学会这个方法少走不少弯路
- JAVA栅栏密码解密程序
- table模板标签,批量多选失效的问题,tr td结构多选框失效相关问题