Claws Garden

使用Groovy实现自己的DSL

这段时间在学习构建工具Gradle,因为Gradle的构建脚本要使用到groovy语言编写,因此就顺便学习了一下groovy。一开始听说gradle脚本其实就是groovy语言的脚本,我是十分震惊的,因为再怎么看都感觉build.gradle都只是像一个配置文件,根本想不到是一段代码。学习了gradle里面的各种特殊的省略写法和闭包的概念之后,才恍然大悟,原来.gradle脚本就是各种省略完了的groovy脚本。

定义成DSL的好处时,让就算没学过Groovy语法的非专业用户也能非常方便直观地,使用接近自然语言的魔法语法来编写代码组织功能。因为听起来真的很cool,我就也尝试自己写了一套用于绘制简单的线段图形的DSL。这个想法来源于《程序员修炼之道》这本书,书中在介绍领域语言的时候引入了一个类似的练习。

目标

使用Groovy定义一套DSL,让下面的脚本可以绘制出直线并展示出来。

 1pad(500, 500) {
 2    color "black"
 3    line {
 4        start(100, 200) end(300, 300)
 5        start(100, 30) end(0, 0)
 6    }
 7    color "red"
 8    line {
 9        start(100, 200) end(100, 300)
10        start(500, 500) end(60, 20)
11    }
12}
13
14pad {
15    size {
16        length = 1000
17        width = 1000
18    }
19    color "red"
20    line {
21        start(100, 200) end(300, 300)
22    }
23}

进行一下简单说明:

  1. pad表示一个新建一个画板,花括号中表示这个画板上的内容
  2. 画板的大小既可以在建立的时候通过圆括号指定,也可以在内容中用size来指定
  3. color表示更换颜色,表示之后的绘图将使用新的颜色。默认黑色
  4. start end表示一条线段开始和结束的坐标,放在line后的块可以定义line

Step0 打桩

直接运行脚本必定报错,首先想办法让脚本不报错。这里打桩的方法是建立一个脚本,用来写自己的DSL实现,然后在现在的类中通过静态引入的方式,将dsl脚本引入。

1// SimpleTest.groovy
2import static top.claws.PaintingDsl.*
3
4pad(500, 500) {
5    color "black"
6    //......
7}

然后在PaintingDsl中,先定义静态的pad函数。

1class PaintingDsl {
2    static def pad(def length, def width, Closure closure) {
3        // TODO
4	}
5    
6    static def pad(Closure closure) {
7        // TODO
8    }
9}

这里牵扯到Groovy中的省略:

这样,pad {...}pad(500, 500) {...} 类似的写法就都不会报错了。

Step1 绘制第一条直线

先不管pad中的内容是怎么样的,想办法可以用java的库来绘制展示第一条直线试试。

 1package top.claws
 2
 3import javax.swing.JFrame
 4import javax.swing.JPanel
 5import java.awt.Color
 6import java.awt.Graphics
 7
 8class PaintingDsl {
 9    def length
10    def width
11
12    PaintingDsl(length, width) {
13        this.length = length
14        this.width = width
15    }
16
17    static def pad(def length, def width, Closure closure) {
18        def dsl = new PaintingDsl(length, width)
19        dsl.show()
20    }
21
22    static def pad(Closure closure) {
23        pad(500, 500, closure)
24    }
25
26    def show() {
27        def panel = new JPanel() {
28            @Override
29            protected void paintComponent(Graphics g) {
30                super.paintComponent(g)
31                g.setColor(Color.RED)
32                g.drawLine(0, 0, 100, 100)
33            }
34        }
35
36        def frame = new JFrame()
37        frame.contentPane = panel
38        frame.setSize(length, width)
39        frame.setLocationRelativeTo(null)
40        frame.defaultCloseOperation = JFrame.EXIT_ON_CLOSE
41        frame.visible = true
42    }
43}

将DSL定义更改成上面这样,就可以绘制第一条红色线段了。

Step2 设定size

给pad设置size应该实现两种方式,第一种就是新建的时候传入大小参数,第二种是在pad的内容中设定。第一种在刚才的部分已经实现了,着重来看第二个:

1pad {
2    size {
3        length = 500
4        width = 500
5    }
6}

一旦出现大括号,就要当做闭包类型的参数,传递给前面的函数。于是这里应该有一个名为size的函数,接收一个闭包参数。之后有一种做法是,使用一个字典成员变量,将闭包中的变量复制全部记录到字典中。

同时要在pad函数中,将closure的delegate指向刚实例化的PaintingDsl对象,让size能够调用到PaintingDsl的成员函数。

 1package top.claws
 2
 3import javax.swing.JFrame
 4import javax.swing.JPanel
 5import java.awt.Color
 6import java.awt.Graphics
 7
 8class PaintingDsl {
 9    def sizeMap = [:]
10
11    PaintingDsl(length, width) {
12        sizeMap.length = length
13        sizeMap.width = width
14    }
15
16    static def pad(def length, def width, Closure closure) {
17        def dsl = new PaintingDsl(length, width)
18        closure.delegate = dsl
19        closure.resolveStrategy = Closure.DELEGATE_ONLY
20        dsl.show()
21    }
22
23    static def pad(Closure closure) {
24        pad(500, 500, closure)
25    }
26
27    def size(Closure closure) {
28        // 等同于设置delegate到sizeMap并执行闭包
29        sizeMap.with closure
30    }
31
32    def show() {
33        def panel = new JPanel() {
34            @Override
35            protected void paintComponent(Graphics g) {
36                super.paintComponent(g)
37                g.setColor(Color.RED)
38                g.drawLine(0, 0, 100, 100)
39            }
40        }
41        
42        // ...
43    }
44    
45    // 下面这两个方法用来占位 避免报错
46    def line(Closure closure) {
47        // TODO
48    }
49    
50    def color(def color) {
51        // TODO
52    }
53}

运行之前的测试脚本,发现其中一个窗口确实变大了。

Step3 添加线条

可以使用一个容器记录所有要添加的线条,然后在pad内容结束后,绘图时,统一添加到panel中。

对于line块的处理,也要使用闭包,而line块中的内容将被作为新闭包的delegate的方法被实现出来。因此这里需要增加一个新的闭包以及新的LineDsl类。同时添加一个Line类方便线段管理。

为了简洁,下面只记录增加的部分。

 1class PaintingDsl {
 2    def lineList = []
 3    
 4    def line(Closure closure) {
 5        def dsl = new LineDsl(this)
 6        closure.delegate = dsl
 7        closure.resolveStrategy = Closure.DELEGATE_ONLY
 8        closure()
 9    }
10    
11    // 更改一下绘图逻辑,绘制列表中的所有线段
12    def show() {
13        def panel = new JPanel() {
14            @Override
15            protected void paintComponent(Graphics g) {
16                super.paintComponent(g)
17                g.setColor(Color.RED)
18                for (def line in lineList) {
19                    g.drawLine(line.xFrom, line.yFrom, line.xTo, line.yTo)
20                }
21            }
22        }
23        // ...
24    }
25}
26
27class LineDsl {
28    def paintingDsl
29    def lineBuilding
30    
31    LineDsl(def paintingDsl) {
32        this.paintingDsl = paintingDsl
33    }
34    
35    def start(x, y) {
36        lineBuilding = new Line()
37        lineBuilding.xFrom = x
38        lineBuilding.yFrom = y
39        this
40    }
41    
42    def end(x, y) {
43        lineBuilding.xTo = x
44        lineBuilding.yTo = y
45        paintingDsl.lineList.add(lineBuilding)
46        lineBuilding = null
47    }
48}
49
50class Line {
51    def xFrom
52    def yFrom
53    def xTo
54    def yTo
55    def color
56}

Step4 处理颜色

只需要有一个成员变量记录当前在用的颜色就可以了。由于传进来的是一个字符串,可以使用一个static的字典记录颜色名称和对应颜色对象的映射关系。

之后再稍微修改LineDsl中end的逻辑,在线段结束的时候标上颜色。然后更改一下绘图的逻辑,遍历线段时对颜色进行重新设置即可。

 1class PaintingDsl {
 2    def curColor = colorMap.black
 3
 4    static colorMap = [
 5            red  : new Color(255, 175, 175),
 6            black: new Color(0, 0, 0),
 7            white: new Color(255, 255, 255)
 8    ]
 9    
10	def color(String color) {
11        curColor = colorMap[color]
12    }
13    
14    def show() {
15        def panel = new JPanel() {
16            @Override
17            protected void paintComponent(Graphics g) {
18                super.paintComponent(g)
19                for (def line in lineList) {
20                    g.setColor(line.color)
21                    g.drawLine(line.xFrom, line.yFrom, line.xTo, line.yTo)
22                }
23            }
24        }
25        // ...
26    }
27}
28
29class LineDsl {
30    def end(x, y) {
31        lineBuilding.xTo = x
32        lineBuilding.yTo = y
33        lineBuilding.color = paintingDsl.curColor // 给线段记录颜色
34        paintingDsl.lineList.add(lineBuilding)
35        lineBuilding = null
36    }
37}

成果展示

完成了所有代码的编写之后,之前的脚本中的需求就全部达成了,效果如下图: image-20230506010132842

这个小实践中涉及的代码:https://gist.github.com/jingjiecb/e3c049ea6e463d9f06a45e879ea0878a

#Groovy