Groovy学习之-Groovy Development Kit(GDK)-实用工具

Groovy学习目录-传送门

ConfigSlurper

ConfigSlurper是一个用于读取以Groovy脚本形式定义的配置文件的实用程序类。 类似于Java * .properties文件的情况,ConfigSlurper允许使用点符号。 但此外,它允许Closure作用域配置值和任意对象类型。

def config = new ConfigSlurper().parse('''
    app.date = new Date()  //点符号的用法
    app.age  = 42
    app {                  //使用Closure作用域作为点符号的替代
        name = "Test${42}"
    }
''')

assert config.app.date instanceof Date
assert config.app.age == 42
assert config.app.name == 'Test42'

从上面的例子可以看出,parse方法可以用来检索groovy.util.ConfigObject实例。ConfigObject是一个专门的java.util.Map实现,它返回配置的值或一个新的ConfigObject实例,但永远不为null

def config = new ConfigSlurper().parse('''
    app.date = new Date()
    app.age  = 42
    app.name = "Test${42}"
''')

assert config.test != null//`config.test`还没有被指定,它被调用时返回一个`ConfigObject`。

点符号是配置变量名称的一部分的情况下,可以使用单引号或双引号对其进行转义。

def config = new ConfigSlurper().parse('''
    app."person.age"  = 42
''')

assert config.app."person.age" == 42

此外,ConfigSlurper还支持environmentsenvironments方法可以用来移交一个本身可能由几个部分组成的Closure实例。 假设我们想为开发环境创建一个特定的配置值。 当创建ConfigSlurper实例时,我们可以使用ConfigSlurper(String)构造函数来指定目标环境。

def config = new ConfigSlurper('development').parse('''
  environments {
       development {
           app.port = 8080
       }

       test {
           app.port = 8082
       }

       production {
           app.port = 80
       }
  }
''')

assert config.app.port == 8080

ConfigSlurper环境不限于任何特定的环境名称。 它只依赖于ConfigSlurper客户端代码支持和解释什么值。

environments方法是内置的,但registerConditionalBlock方法可用于注册除environments名称之外的其他方法名称。

def slurper = new ConfigSlurper()
slurper.registerConditionalBlock('myProject', 'developers')   //一旦新block被注册,ConfigSlurper可以解析它。

def config = slurper.parse('''
  sendMail = true

  myProject {
       developers {
           sendMail = false
       }
  }
''')

assert !config.sendMail

对于Java集成,toProperties方法可用于将ConfigObject转换为可能存储到* .properties文本文件的java.util.Properties对象。 请注意,将配置值添加到新创建的Properties实例时,将其转换为String实例。

def config = new ConfigSlurper().parse('''
    app.date = new Date()
    app.age  = 42
    app {
        name = "Test${42}"
    }
''')

def properties = config.toProperties()

assert properties."app.date" instanceof String
assert properties."app.age" == '42'
assert properties."app.name" == 'Test42'

Expando

Expando类可用于创建动态可扩展对象。 尽管它的名称叫Expando,它不使用类ExpandoMetaClass。 每个Expando对象表示一个独立的动态制作的实例,可以在运行时使用属性(或方法)进行扩展。

def expando = new Expando()
expando.name = 'John'

assert expando.name == 'John'

当动态属性注册Closure代码块时,会发生特殊情况。 一旦Closure被注册,它就可以被调用,因为它将通过方法调用完成。

def expando = new Expando()
expando.toString = { -> 'John' }
expando.say = { String s -> "John says: ${s}" }

assert expando as String == 'John'
assert expando.say('Hi') == 'John says: Hi'

可观察的 list, map and set

Groovy带有可观察的 list, map and set。 当添加,删除或更改元素时,每个集合都会触发java.beans.PropertyChangeEvent事件。 注意,PropertyChangeEvent不仅发出某个事件已经发生的信号,此外,它保存关于属性名称和某个属性已经改变到的旧/新值的信息。

根据发生的更改的类型,可观察的集合可能触发更专门的PropertyChangeEvent类型。 例如,向可观察列表中添加元素会触发ObservableList.ElementAddedEvent事件。

def event                                       //声明一个PropertyChangeEventListener捕获被触发的事件
def listener = {
    if (it instanceof ObservableList.ElementEvent)  {  //ObservableList.ElementEvent及其子类型与此侦听器相关
        event = it
    }
} as PropertyChangeListener


def observable = [1, 2, 3] as ObservableList    //注册监听器
observable.addPropertyChangeListener(listener)  //从给定列表创建一个ObservableList

observable.add 42                               //触发ObservableList.ElementAddedEvent事件

assert event instanceof ObservableList.ElementAddedEvent

def elementAddedEvent = event as ObservableList.ElementAddedEvent
assert elementAddedEvent.changeType == ObservableList.ChangeType.ADDED
assert elementAddedEvent.index == 3
assert elementAddedEvent.oldValue == null
assert elementAddedEvent.newValue == 42

请注意,添加元素实际上会触发两个事件。 第一个是类型 ObservableList.ElementAddedEvent,第二个是一个普通的PropertyChangeEvent,通知监听器属性size的更改。

ObservableList.ElementClearedEvent事件类型是另一个有趣的事件类型。 每当删除多个元素时,例如调用clear()时,它保存从列表中删除的元素。

def event
def listener = {
    if (it instanceof ObservableList.ElementEvent)  {
        event = it
    }
} as PropertyChangeListener


def observable = [1, 2, 3] as ObservableList
observable.addPropertyChangeListener(listener)

observable.clear()

assert event instanceof ObservableList.ElementClearedEvent

def elementClearedEvent = event as ObservableList.ElementClearedEvent
assert elementClearedEvent.values == [1, 2, 3]
assert observable.size() == 0

为了获得所有支持的事件类型的概述,我们鼓励读者查看JavaDoc文档或正在使用的observable集合的源代码。

ObservableMapObservableSet带有我们在本节中对ObservableList所看到的相同的概念。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,107评论 19 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 47,149评论 6 342
  • 导语: 随着技术的发展,不管是前端开发、服务端开发或者是移动端开发(移动也是前端的一个分支)中都会用到自动化构建工...
    伊始雨深阅读 8,175评论 0 4
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 33,300评论 18 399
  • 1. 简介 1.1 什么是 MyBatis ? MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的...
    笨鸟慢飞阅读 11,207评论 0 4

友情链接更多精彩内容