Groovy高效编程——动态改变对象的能力(1)

从Groovy1.1beta-2开始,实现动态改变对象的能力变的十分简单:

一开始,我们有这样一个类:

class Person {
String name
}

该类的实例都是哑巴,不能说话,作为造物主的我们该完善它们,使它们能自我介绍(添加实例方法):

class Person {
String name
}

// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}

现在让我们看看,它们到底是否真的能够开口自我介绍了呢:

class Person {
String name
}

// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}
def person = new Person(name:"山风小子")
person.introduce()

运行结果:
I'm 山风小子

嗯~人类改造成功~

但人应该有性别吧,嗯~对的,加个性别属性sex(添加属性):

class Person {
String name
}

// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}
// 添加性别属性认为男(Male)
Person.MetaClass.sex = "Male"
def person = new Person(name:"山风小子")
person.introduce()
println person.sex

运行结果:
I'm 山风小子
Male


但做男人累啊~为了买房,娶妻拼命赚钱,做女人算了,做变性手术:

class Person {
String name
}

// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}
// 添加性别属性认为男(Male)
Person.MetaClass.sex = "Male"
def person = new Person(name:"山风小子")
person.introduce()
println person.sex
// 做变性手术,变为女的(Female)
person.sex = "Female"
println person.sex

运行结果:
I'm 山风小子
Male
Female

作为造物主的我们考虑到手术的风险性,为了让其他人知道自己现在是个女的,在介绍中添加性别说明:

class Person {
String name
}

// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}
// 添加性别属性认为男(Male)
Person.MetaClass.sex = "Male"
// 修改之前自我介绍行为,添加性别说明
Person.MetaClass.introduce << {println "I'm $name,$sex"}
def person = new Person(name:"山风小子")
person.introduce()
// 做变性手术,变为女的(Female)
person.sex = "Female"
person.introduce()

运行结果:
I'm 山风小子,Male
I'm 山风小子,Female

为了造人方便点,搞个工厂方法添加方法,即静态方法):

class Person {
String name
}
// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}
// 添加性别属性认为男(Male)
Person.MetaClass.sex = "Male"
// 修改之前自我介绍行为,添加性别说明
Person.MetaClass.introduce << {println "I'm $name,$sex"}
def person = new Person(name:"山风小子")
person.introduce()
// 做变性手术,变为女的(Female)
person.sex = "Female"
person.introduce()
// 工厂方法,造人方便点
Person.MetaClass.'static'.createPerson = { name,sex ->
Person p = new Person()
p.name = name
p.sex = sex
return p
}
def bluesun = Person.createPerson("山风小子","Male")
bluesun.introduce()

运行结果:
I'm 山风小子,Male
I'm 山风小子,Female
I'm 山风小子,Male

为了方便实例化Person,添加一个构造方法添加构造方法):

class Person {
String name
}
// 添加自我介绍的行为
Person.MetaClass.introduce << {println "I'm $name"}
// 添加性别属性认为男(Male)
Person.MetaClass.sex = "Male"
// 修改之前自我介绍行为,添加性别说明
Person.MetaClass.introduce << {println "I'm $name,"Male")
bluesun.introduce()
// 方便实例化Person,添加一个构造方法
Person.MetaClass.constructor << { name,sex ->
new Person(name:name,sex:sex)
}
def daniel = new Person("Daniel","Male")
daniel.introduce()

运行结果:
I'm 山风小子,Male
I'm Daniel,Male

最后,引用一个官方例子swapCase来展示一下Groovy是如何增强既有类的能力的(演示如何使用delegate,注意演示的是final类:String)

String.MetaClass.swapCase = {->
def sb = new StringBuffer()
// delegate与this类似,引用当前正被‘改造’的对象
delegate.each {
sb << (Character.isUpperCase(it as char) ? Character.toLowerCase(it as char) :
Character.toupperCase(it as char))
}
sb.toString()
}

String s = "Hello,world!"
println s.swapCase()

运行结果:
hELLO,WORLD!

<<用于添加方法(如果方法已经存在,会发生groovy.lang.GroovyRuntimeException异常),=用于添加方法或覆盖既有方法

相关文章

背景:    8月29日,凌晨4点左右,某服务告警,其中一个...
https://support.smartbear.comeadyapi/docs/soapui/steps/g...
有几个选项可用于执行自定义JMeter脚本并扩展基线JMeter功能...
Scala和Java为静态语言,Groovy为动态语言Scala:函数式编程,...
出处:https://www.jianshu.com/p/ce6f8a1f66f4一、一些内部...
在运行groovy的junit方法时,报了这个错误:java.lang.Excep...