文章永久连接:https://tech.souyunku.com/?p=8485
元对象编程或MOP可以用于动态调用方法,并且可以即时创建类和方法。
那么这是什么意思呢?让我们考虑一个叫Student的类,它是一个没有成员变量或方法的空类。假设你必须在这个类上调用以下语句。
Def myStudent = new Student()
myStudent.Name = ”Joe”;
myStudent.Display()
现在在元对象编程中,即使类没有成员变量Name或方法Display(),上面的代码仍然可以工作。
这如何工作?那么,为了这个工作,一个人必须实现GroovyInterceptable接口挂钩到Groovy的执行过程。以下是该接口的可用方法。
Public interface GroovyInterceptable {
Public object invokeMethod(String methodName, Object args)
Public object getproperty(String propertyName)
Public object setProperty(String propertyName, Object newValue)
Public MetaClass getMetaClass()
Public void setMetaClass(MetaClass metaClass)
}
所以在上面的接口描述中,假设你必须实现invokeMethod(),它会被调用的每个方法,要么存在或不存在。
缺失属性
所以,让我们看一个例子,我们如何为缺失的属性实现元对象编程。以下键应该注意以下代码。
- 类Student没有定义名为Name或ID的成员变量。
- 类Student实现GroovyInterceptable接口。
- 有一个称为dynamicProps的参数,将用于保存即时创建的成员变量的值。
- 方法getproperty和setproperty已被实现以在运行时获取和设置类的属性的值。
class Example {
static void main(String[] args) {
Student mst = new Student();
mst.Name = "Joe";
mst.ID = 1;
println(mst.Name);
println(mst.ID);
}
}
class Student implements GroovyInterceptable {
protected dynamicProps=[:]
void setProperty(String pName,val) {
dynamicProps[pName] = val
}
def getProperty(String pName) {
dynamicProps[pName]
}
}
以下代码的输出将是 –
Joe
1
缺失方法
所以,让我们看一个例子,我们如何为缺失的属性实现元对象编程。以下键应该注意下面的代码 –
- 类学生现在实现invokeMethod方法,它被调用,而不管该方法是否存在。
class Example {
static void main(String[] args) {
Student mst = new Student();
mst.Name = "Joe";
mst.ID = 1;
println(mst.Name);
println(mst.ID);
mst.AddMarks();
}
}
class Student implements GroovyInterceptable {
protected dynamicProps = [:]
void setProperty(String pName, val) {
dynamicProps[pName] = val
}
def getProperty(String pName) {
dynamicProps[pName]
}
def invokeMethod(String name, Object args) {
return "called invokeMethod $name $args"
}
}
以下代码的输出如下所示。请注意,即使方法Display不存在,也没有缺少方法异常的错误。
Joe
1
元类
此功能与MetaClass实现相关。在默认实现中,您可以访问字段而不调用它们的getter和setter。以下示例显示如何通过使用metaClass函数,我们能够更改类中的私有变量的值。
class Example {
static void main(String[] args) {
Student mst = new Student();
println mst.getName()
mst.metaClass.setAttribute(mst, 'name', 'Mark')
println mst.getName()
}
}
class Student {
private String name = "Joe";
public String getName() {
return this.name;
}
}
以下代码的输出将是 –
Joe
Mark
方法缺失
##
Groovy支持methodMissing的概念。此方法与invokeMethod的不同之处在于,它仅在失败的方法分派的情况下被调用,当没有找到给定名称和/或给定参数的方法时。以下示例显示如何使用methodMissing。
class Example {
static void main(String[] args) {
Student mst = new Student();
mst.Name = "Joe";
mst.ID = 1;
println(mst.Name);
println(mst.ID);
mst.AddMarks();
}
}
class Student implements GroovyInterceptable {
protected dynamicProps = [:]
void setProperty(String pName, val) {
dynamicProps[pName] = val
}
def getProperty(String pName) {
dynamicProps[pName]
}
def methodMissing(String name, def args) {
println "Missing method"
}
}
以下代码的输出将是 –
Joe
1
Missing method