Groovy探索 Visitor模式
Groovy语言中的Visitor模式的实现,其实也跟闭包有很大的、直接的关系。当然,你也可以完全不用闭包来实现Visitor模式,就像Java语言那样来实现。但使用了闭包的Visitor模式更加简单,更加易于扩展。所以,我们在Groovy语言中,倾向于使用闭包来实现Visitor模式。这也是我很久以来,都想把这篇文字放在《Groovy探索 闭包》系列中的原因,但由于Visitor模式的复杂性,使得我倾向于以模式的观点来说明它,以利于我们的理解。
Java语言实现的Visitor模式一向是难于理解的一个模式,它的实现基本上有两大类的接口,众多的实体实现。在这里我不多说,有兴趣的看官或者没有学过Java语言Visitor模式的,可以在网上找资料来看。
其实,Visitor模式的目的非常简单,就是把被访问者的数据结构和访问者的动作分离,有利于访问者的扩展。这里,值得注意的是:该模式强调的是被访问者要有一定的确定性,访问者才是可以扩展的。
在Groovy语言中,我们的访问者被简化成一个个的闭包,不再需要一个Visitor接口,让我们从一个简单得不能再简单的例子说起。
比如,我们有如下的一个被访问者对象:
def visitable = [1,2,3,4,5]
这是一个简单的数组对象,现在,我们的第一个访问者希望打印出被访问者的所有元素来。我们就可以这样来定义一个访问者:
def visitor1 = {
println it
}
现在,我们让访问者去访问被访问者:
visitable.each(visitor1)
运行结果为:
1
2
3
4
5
当然,我们使用Visitor模式的目的,就是要扩展访问者,现在,我们就可以扩展另外一个访问者了。
def total = 0
def visitor2 = {
total += it
}
我们也让这个访问者来访问被访问者:
visitable.each(visitor2)
println total
运行结果为:
15
通过上面的例子,我们可以看到,在使用Groovy语言实现的Visitor模式中,我们是如何使用闭包来实现Visitor接口的实现的。正是通过闭包,我们在Groovy语言实现的Visitor模式中,就再也找不到Visitor接口了。
现在,我们再理解Java语言实现的Visitor模式的Visitor接口,其实就是让访问者动作有一个实体类可以存在,因为在Java语言中,方法是不能独立存在的。而在Groovy语言中,方法能够以闭包的形式独立存在,所以就用不着Visitor接口了。
理解了这一层关系以后,我们上面的代码就可以进一步简化成如下的样子了:
visitable.each{
println it
}
visitable.each{
total += it
}
println total
在这里,我们连声明一个Visitor对象都省了,但它们实现的也是Visitor模式。
上面是一个最简单不过的例子,不能显示出Visitor模式的复杂性来。下面,我们就来说说一个相对复杂一点的例子来,进一步的分析Visitor模式。
这个例子说的是,我们在优惠期间到电脑店去买电脑及周边产品。电脑店在优惠期间是这样买商品的:电脑打九折,光盘是买5送1。现在,我们要买一台电脑,12张光盘,想知道我们花了多少钱。
现在,我们就来模拟这个场景。
首先,我们定义的是被访问者和访问者之间的接口类:
class Goods {
def accept(Closure doAction)
{
doAction(this)
}
}
这个Goods类的目的很简单,就是让访问者对象能够访问到被访问者对象,如果没有这个Goods类,那么上面的"accept"方法就要分布在各个被访问者类中。而"accept"方法中的输入参数"doAction",就是访问者对象了。
现在,该我们的Computer类出场了:
class Computer extends Goods{
def amount
def price
def static DISCOUNT = 0.9
def totalPrice()
{
price*amount*DISCOUNT
}
def totalDiscount()
{
price*amount*(1-DISCOUNT)
}
}
它也很简单:有两个输入属性分别用来定义购买的数量和单价,一个常量属性用来定义折扣。两个方法分别来获取总价和总的折扣价。
同样的功能是Disk类,现在由它出场:
class Disk extends Goods{
def amount
def price
def totalPrice()
{
(amount-(amount/5 as int))*price
}
def totalDiscount()
{
(amount/5 as int)*price
}
}
它实现的是买五送一的活动。下面的Sales类来定义我们买了多少东西:
class Sales {
List goods
def accept(Closure doAction)
{
goods.each{
it.accept(doAction)
}
}
}
它的"accept"方法用来使得访问者能够访问我们所买的所有东西。当然,该方法的"doAction"就是我们的访问者对象了。
现在,一切就绪,我们就可以来访问被访问者了。
首先,我们想知道我们总共花了多少钱?
def mySales = new Sales(goods:[new Computer(amount:1,price:500),new Disk(amount:12,price:2)])
上面的初始化,我们共买了单价为500的电脑一台,单价为2的光盘12张。我们定义一个变量来存取我们所花的钱:
def totalPrice = 0
我们来计算我们总共花了多少钱:
mySales.accept{
totalPrice += it.totalPrice()
}
println "my total price is ${totalPrice}"
结果为:
my total price is 470.0
这样,一个稍微复杂一点的Visitor模式就全部实现了。
你可能要问了,我的访问者对象不是可以扩展吗?你扩展一个给我看看,好,我们现在来计算我们总共省了多少钱。
def totalDiscount = 0
mySales.accept{
totalDiscount += it.totalDiscount()
}
println "my total discount is ${totalDiscount}"
运行结果为:
my total discount is 54.0
使用闭包实现的Visitor模式还有一个好处是能够扩展被访问者,使用在Java语言中实现的Visitor所难以做到的,如果要做到,一般要用到反射技术。而在Groovy语言闭包所实现的Visitor中,我们可以很轻松的扩展被访问者。
比如,我们现在还想买一个鼠标,它没有折扣,如下:
class Mouse extends Goods{
def amount
def price
def totalPrice()
{
amount*price
}
def totalDiscount()
{
0
}
}
现在,我们的客户端就可以这样使用:
def mySales = new Sales(goods:[new Computer(amount:1,price:500),new Disk(amount:12,price:2),
new Mouse(amount:1,price:10)])
def totalPrice = 0
mySales.accept{
totalPrice += it.totalPrice()
}
println "my total price is ${totalPrice}"
def totalDiscount = 0
mySales.accept{
totalDiscount += it.totalDiscount()
}
println "my total discount is ${totalDiscount}"
运行结果为:
my total price is 480.0
my total discount is 54.0
可以看出,在Groovy语言中使用闭包实现的Visitor是非常好理解的,简单到你没有意识到你是在使用Visitor模式,而且,这个Visitor模式也很容易扩展它的被访问者。就像上面的例子那样。
分享到:
相关推荐
例如,Groovy可以很自然地实现Visitor模式和Builder模式等。 文档最后对Groovy语言进行了整体的概述,强调了Groovy是一种强大、高效且富有表达力的编程语言,适合多种不同的开发场景和开发人员群体,从简单的脚本...
apache-groovy-3.0.8.zip apache官网的groovy3.0.8版本,希望大家多多下载,apache-groovy-3.0.8.zip apache官网的groovy3.0.8版本,希望大家多多下载,apache-groovy-3.0.8.zip apache官网的groovy3.0.8版本,希望...
1. 引入Groovy库:在Java项目中添加Groovy的相关依赖,通常是`groovy-all`,确保Java能够访问Groovy运行时环境。 2. 创建GroovyClassLoader:使用这个类加载器可以动态加载和执行Groovy脚本。它继承自Java的...
3. **闭包支持**:闭包是Groovy的一个强大特性,它可以像对象一样被传递和存储,用于实现高级编程模式。 4. **元编程**:Groovy支持元编程,可以通过反射机制修改程序的行为。 5. **简洁的语法**:Groovy具有简洁的...
Groovy SDK 4.0.3 是一个针对Groovy编程语言的软件开发工具包,它包含了Groovy语言的运行环境和开发所需的各种组件。...通过安装和配置这个SDK,你可以开始探索Groovy的动态世界,体验它在各种应用场景中的强大功能。
Groovy是一种动态、开源的编程语言,它是Java平台上的一个JVM(Java Virtual Machine)语言。Groovy结合了Python、Ruby和Perl等脚本语言的简洁性和灵活性,并且完全兼容Java,可以无缝地与Java代码集成。在"groovy-...
Groovy是一种动态、灵活的编程语言,它是Java平台上的一个扩展,可以无缝集成到Java项目中。Groovy的语法简洁,支持面向对象编程、函数式编程,并提供了许多现代语言特性,如闭包和动态类型。这使得Groovy成为快速...
《Groovy in Action》这本书由Dierk König与Andrew Glover、Paul King、Guillaume Laforge以及Jon Skeet共同编写,是一本深入探索Groovy编程语言的权威指南。本书由Manning Publications出版,全面覆盖了Groovy语言...
Groovy是一种基于Java平台的、动态的、强大的编程语言,它设计的目标是增强开发者的生产力。Groovy结合了Java的静态类型系统和Python、Ruby等动态语言的灵活性,使得开发者可以更加高效地编写代码。本篇文章将深入...
Groovy 入门教程 Groovy 是一种基于 Java 语言的脚本语言,运行在 JVM 中,语法与 Java 相似,但抛弃了 Java 的一些烦琐的语法规则,提供了更加简洁和灵活的编程体验。 Groovy 的特点 1. 简洁的语法:Groovy 语法...
Groovy是一种动态、开源的编程语言,它是Java平台上的一个重要的补充。Groovy结合了Python、Ruby和Smalltalk等语言的特性,同时保留了与Java的无缝集成能力,使得它在脚本编写、Web开发、自动化测试等领域有着广泛的...
标题中的“Groovy和Java相互调用1”指的是在编程时如何在Groovy语言环境中调用Java类,以及反之,如何在Java程序中调用Groovy类。这是一种跨语言交互的方式,特别是在混合使用Groovy和Java的项目中非常常见。 ...
### Eclipse 安装 Groovy 插件的详细步骤与使用指南 #### 一、Groovy 插件的安装 为了能够在 Eclipse 中使用 Groovy 进行开发,首先需要安装 Groovy 插件。以下是详细的安装步骤: 1. **下载 GroovyEclipse.zip ...
Grails是基于Groovy的全栈Web应用框架,它简化了MVC开发模式,提供了丰富的插件系统和自动代码生成功能。与Java的Spring Boot框架结合使用时,Groovy可以作为Spring Boot的源代码语言,提供更加简洁的配置和更快的...
### Groovy速查手册知识点详解 #### 一、Groovy简介与特性 Groovy是一种为Java虚拟机(JVM)设计的动态语言。它具备完全的对象导向性、可选的类型系统、操作符定制能力、简洁的数据类型声明、闭包(Closures)、...
- **Builder模式**:Groovy通过Builder模式简化了XML和其他结构化数据的生成。 - **GDK(Groovy Development Kit)**:GDK扩展了Java类库的功能,为常见任务提供了更简洁的API。 - **数据库编程**:Groovy提供了...
Apache Groovy SDK 4.0.1 是一个重要的软件开发工具包,专为使用Groovy编程语言进行开发的程序员设计。Groovy是一种基于Java平台的动态、灵活的编程语言,它扩展了Java语言的功能,提供了简洁的语法和强大的元编程...