1.1、问题域与解答域

1.1.1、什么是DSL

DSL(Domain-Specific Language)全称领域专用语言,就是专门用户特定领域的语言,看着概念觉得挺高大上的,其实很简单,就是专门用于某一个领域使用的语言。举个例子,我们在设置页面样式时,就经常使用CSS样式,那么这CSS语言就是在网页样式设计这一领域专门使用的语言,CSS就是一种DSL;同理,在数据库中使用的SQL,只能在数据库中使用,也是一种DSL。在问题域里面复杂的业务,在解答域中使用DSL语言应该能得到恰当的实现。

1.1.2、那么什么是问题域与解答域?

我们一般把分析业务的那些过程、实体和约束条件称为问题域。业务越复杂,问题域就越复杂。

问题域的分析模型是用解答域提供的工具和手段实现出来的,问题域的元素需要映射成解答域中十点过的技术手段,比如面向对象中的类、对象和方法就是解答域的基本组件。你可以把这些组件组合起来,成为更大的组件。

1.2、DSL与通用编程语言的区别

DSL语言只需要针对一个特定的问题领域就可以了;用DSL编程时只需要处理问题域的复杂性(业务的复杂性),你不用管解答域具体是怎么实现的,因为都使用一定的抽象层次封装起来了;DSL还能够做到给用户使用和阅读(这要求开发者在DSL实现时抽象要合理)

1.3、初窥DSL

1.3.1、DSL对业务用户来说有什么好处?

上面提到DSL用户都能阅读,这样对酬金来说,政策配置的有没有问题,业务用户可以自己去阅读和修改,这是非DSL做不到,比如下面四川酬金中配置的分段条件是用C语言来配置的,用户谁能看懂啊,没有编程经验的估计都看不大懂,这里我给大致解释下下面代码的含义,逻辑复杂度一致即可,并不是实际的业务——“当该业务正在打折时,总金额为该业务笔数(v_dci1 比如从数据库中查询得知)乘以折后单价;如果没打折,为业务笔数乘以折前单价”,是不是觉得中文这么简单的一句话,用现在的C实现是不是太复杂了。

假如我们自己搞一个DSL,这个分段条件可以配置成什么样呢?比如可以这样“When $1 is ‘Y’ then return v_dci1*0.28 else return v_dci1*0.4”,哇,就这么一句就实现了上面13行的逻辑,是不是觉得超简单超清晰,DSL就是这么牛逼。你可能问底层是如何实现的,这底层你可以借助其他牛逼的语言来搞,也可以自己实现语法树来实现(看着难,其实有工具的,别怕),这都是后面要讲的。
上面的例子中可以总结出两点DSL对业务用户的好处:1、DSL给用户更高层次的抽象,不必去关心底层的细节,专注于解决手头的委托;2、DSL只提供有限的语汇,不超出他所描述的领域范围。
1.3.2、目前流行的DSL
之前有提到用于关系型数据库的SQL、用于样式描述的CSS、用于Web标记语言的HTML,书中还有其他的不太认识,这里就不列举了。
使用DSL的话 业务用户也能看懂,这样拉近了开发人员与领域专家(业务专家)的距离,领域专家看的懂,那么他就可以理解解答域的抽象都实现了哪些业务规则,以及这些业务规则实现的全不全面、有没有漏洞
1.3.3、DSL的结构(重要)
设计得当的DSL应该体现以下3个原则,这样才能与领域用户更好的“沟通”,让用户能够看懂和编写DSL:
1、DSL要为问题域制品提供直接的映射,如酬金问题域有一个名为Trade交易的实体,那么DSL脚本就必须包含同样名称同样角色的一个抽象概念;
2、DSL脚本必须使用问题域的共通语汇。比如用户一个渠道概念,开发人员一个技术上的组织树的Node节点概念,那么可以试着用“网点”这个概念来替代,这样大家都容易理解(这里只是举个例子,可能不大恰当)
3、DSL脚本必须对底层实现进行抽象。不能把底层实现暴露给用户,这样会增加复杂度,尽量合适的高度抽象,去除因为实现细节而引入的非本质的复杂性。
1.4、DSL执行方式
1、有的可以直接运行脚本执行,如awk、sed可以直接执行DSL
2、在虚拟机上开发的DSL脚本虚拟机上执行。任何Java DSL脚本的语义模型都生成在JVM上执行的字节码。
3、有些语言提供编译时元编程能力。在编译后运行前就可以变成一般的语法结构,就可以直接执行了。
1.5、DSL的分类
根据是否使用已有的语言作为宿主语言,在该语言上开发DSL来分,分为两类:
1、内嵌式DSL(内部DSL):使用了其他语言作为宿主语言,将DSL的实现嵌入到了宿主语言之中,称为内部DSL;
2、独立DSL(外部DSL):不依赖于任何现有语言创立的DSL。如果你了解JVM基础,那么你知道从零搞出来的语言需要设计以下功能:词法分析、解析技术、解释、编译、执行,从零实现DSL复不复杂看你要支撑到什么程度,而且还有工具帮助你
1.6、DSL的优缺点
优点:
1、使用领域高度抽象的概念,更具表现力,用户更喜欢,跟C实现的分段条件一比就知道了
2、DSL更精炼
3、基于更高的抽象层次,弱化底层实现
4、从长远来说回报更大
5、团队自己实现的语言,更容易扩展
缺点:
1、即使将DSL寄生于已有的高级语言,但是设计一门语言还是挺有难度的,视角一定要足够高,抽象要足够高才行
2、前期投入大,跟不用DSL相比,前期投入肯定大,但是从整个阶段来看还是值得的。
3、可能带来性能隐忧,因为他毕竟多了一层语言
4、可能缺乏足够的工具支持,如可能没有带代码高亮、智能提示的IDE开发工具、单元测试支持、性能分析工具等
5、多学一门语言的成本,外部DSL就要学该DSL,内部DSL还需要掌握宿主语言。
6、使用了DSL语言可能与其他现有的DSL有冲突(我们应该遇不到)
总结:
从上面看,如果要设计一门DSL,抽象层级一定要足够高,这就最好需要领域专家的参与和把关。
 
http://www.sunyaozong.com/what-is-dsl/

《DSL》笔记一、什么是DSL(转)的更多相关文章

  1. Android开发:《Gradle Recipes for Android》阅读笔记(翻译)6.2——DSL文档

    问题: 你需要查找Android Gradle DSL的完整文档. 解决方案: 访问Gradle Tools网站,从Android开发网站下载ZIP文件. 讨论:Android开发网站首页有完整的AP ...

  2. DSL简介(转)

    DSL编程:有人将DSL编程称之为声明式(Declarative)编程.DSL是在模型之上建立的一种更加灵活的对 模型化的理解和使用方式.语义模型是DSL的核心.内部DSL:用通用语言的语法表示DSL ...

  3. Atitit dsl exer v3 qb3 新特性

    Atitit dsl exer v3 qb3 新特性 /atiplat_cms/src/com/attilax/dsl/DslParser.java V3 支持typeed参数,与简化的notyp参数 ...

  4. Elasticsearch(入门篇)——Query DSL与查询行为

    ES提供了丰富多彩的查询接口,可以满足各种各样的查询要求.更多内容请参考:ELK修炼之道 Query DSL结构化查询 Query DSL是一个Java开源框架用于构建类型安全的SQL查询语句.采用A ...

  5. Atitit. 提升软件开发效率and 开发质量---java 实现dsl 4gl 的本质and 精髓 O725

    Atitit. 提升软件开发效率and 开发质量---java 实现dsl 4gl 的本质and 精髓  O725 1. DSL主要分为三类:外部DSL.内部DSL,以及语言工作台. 1 2. DSL ...

  6. Paip.声明式编程以及DSL 总结

    Paip.声明式编程以及DSL 总结     1.1      声明式编程DSL 1.2      声明式语言) 1.3      声明式编程框架AOP实现 1.4      应用场合 1.5     ...

  7. 使用Antlr实现简单的DSL

    为什么要使用DSL DSL是领域专用语言,常见的DSL有SQL,CSS,Shell等等,这些DSL语言有别于其他通用语言如:C++,Java,C#,DSL常在特殊的场景或领域中使用.如下图: 领域专用 ...

  8. 简述 Ruby 与 DSL 在 iOS 开发中的运用

    阅读本文不需要预先掌握 Ruby 与 DSL 相关的知识 何为 DSL DSL(Domain Specific Language) 翻译成中文就是:"领域特定语言".首先,从定义就 ...

  9. BDD 与DSL 入门

    正文: Behavior Driven Development,行为驱动开发是一种敏捷软件开发的技术,它鼓励软件项目中的开发者.QA和非技术人员或商业参与者之间的协作.在了解Behavior Driv ...

随机推荐

  1. 网关/负载均衡下的consul集群代理

    之前有做过使用单机版的consul实现Prometheus服务注册,以为使用集群版的consul只是将consul服务地址从节点IP变为了网关IP.但比较坑的就是,当使用consul注册一个servi ...

  2. 鼠标滚轮滚动慢/拖动Office出现滞后问题处理

    一.说明 我对外设一直不是很了解,买的鼠标今天到了,使用时遇到了两个问题在这里记一下. 二.滚轮滚动慢处理 问题描述:在网页中滚动滚轮每次只能上下移动一点点,感觉很难受. 问题原因:此问题是滚轮滚动一 ...

  3. NVDLA软件架构和源码解析 第一章—内核驱动【华为云技术分享】

    驱动整体设计介绍 不同的processor Nvidia DLA的内核驱动KMD(Kernel mode driver)中,并不是把DLA当成一个设备来控制,而是把不同的功能模块当做不同的proces ...

  4. Linux中常用命令pipe

    大多数linux命令处理数据后都会输出到标准输出,但是如果数据要经过系列列的步骤处理后,才是需要的数据个数,这种需求就需要管道来帮助完成. 管道命令使用"|"作为界定符,将界定符前 ...

  5. Python列表(list)所有元素的同一操作

    针对很普遍的每个元素的操作会遍历每个元素进行操作. 这里给出了几种写法,列表每个元素自增等数学操作同理: 示例:整形列表ilist加1个数.元素类型转字符串: ilist = [1, 2, 3, 10 ...

  6. Asp.Net Core中使用GDI+绘图提示gdiplus库找不到的问题

    参考  https://www.cnblogs.com/VirtualMJ/p/9917916.html 文章中   1 2 3 yum install -y epel-release yum mak ...

  7. javascript query string

    function getQueryVariable(variable) { var query = window.location.search.substring(1); var vars = qu ...

  8. Mybatis映射器接口代理对象的方式 运行过程

    查询一张表的所有数据. 环境: 使用工具IntelliJ IDEA 2018.2版本. 创建Maven工程不用骨架 1.pom.xml <?xml version="1.0" ...

  9. [Linux] Ubuntu Server18 python3.7 虚拟环境

    Ubuntu Server18 python3.7 环境 Ubuntu Server18 默认是python3.6, 目前开发主要用python3.7. 所以想搭建python3.7环境. 试过几手动 ...

  10. DOM创建节点

    1.DOM--document object model 常用的节点类型: 元素节点:(标签) 属性节点:(标签里的属性) 文本节点:(文本节点) 2,document有个属性叫nodetype,返回 ...