首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 其他教程 > 其他相关 >

groovy闭包跟函数(转)

2012-10-27 
groovy闭包和函数(转)一、代码块在代码块中用def关键字定义的变量不能在外部访问,如: try{def msgHello,w

groovy闭包和函数(转)
一、代码块
在代码块中用def关键字定义的变量不能在外部访问,如:

try{
  def msg="Hello,world!";
}
//pringln msg;

如果运行最后一行"println msg;",则会报错。

但是,在代码块中没有用def关键字声明的变量,就可以在外部访问,如:

try{
msg="Hello,world!";
}
println msg;

已经使用def定义的变量,不能再次用def定义:
def a=10;
//def a=20;


代码块是可以嵌套的:

try{
try{
  try{
   println "Hello,world!";
  }
}
}

二、闭包
闭包是可以引用外部上下文环境的一系列语句。可以将闭包赋予一个变量,并在稍后执行。

每个闭包都有一个返回值,默认的返回值就是该闭包中最后一行语句的结果

闭包可以访问外部的变量
def x=1;

def b={
x*2;//闭包可以访问外部的变量
}

将闭包赋予一个变量将记住该闭包创建时的上下文环境,即使运行时已经超出了原来的作用域:
def c;

try{
def m=10;
c={m};
}

println c();
在此处调用c(),即使已经超出了block的作用域,但仍能输出m的值


将一个闭包放入另一个闭包,可以创建它的两个实例:

c={e={"Hello,World";};e;}
v1=c();
v2=c();
println v1==v2//在此得出false,说明c()返回了不同的实例

可以在定义闭包时声明一些参数,之后在调用闭包时将这些参数值传入闭包:

c={n-> println "Parameter="+n;}

c(10);

可以利用参数将信息从包内导出,也就是所谓的“输出参数”

c={o,p-> o<<p};
x=[];
c(x,1);
c(x,2);
c(x,3);
println x;

在闭包中,总是可以用到一个叫做it的变量,如果没有显式的定义参数,则可以用it来引用

c={"Parameter=${it}"};
println c(10);

如果没有传入任何参数,it仍然存在,但是为null

闭包的参数不能使用作用域中已存在变量的名字,除了隐式变量it:
def it=10;

c={it*it};

println c(5);//在此,隐式变量it仍然引用参数5,而非外部定义的10

参数在调用闭包时可以被命名。这些被命名的参数将被组装进一个Map,并指定给闭包的第一个参数:
def x={a,b,c->a.m+a.n+a.o-b+c;};

println x(m:10,5,n:15,6,o:20);

在调用闭包x的时候,参数m:10,n:15,o:20被组装进Map赋予形参a,5、6按顺序分别赋予形参b和c,结果是46。

我们可以在闭包内部和外部查询参数的个数:


def c={a,b,c->getMaximumNumberOfParameters();}

println c(1,2,3);

println c.getMaximumNumberOfParameters();

调用闭包的时候,可以对最后的一个或几个参数应用默认值:

def c={a,b=2,c=3->a+b+c;};

println c(1);

结果输出6

一个闭包可以在最后一个参数前添加Object[]前缀来创建可变长度的参数:

def c={a,Object[] b->a+b[0]+b[1];}

println c(1,2,3);

我们可以使用一个list参数调用闭包。如果闭包没有定义一个list类型的参数,则此参数将作为分解的参数传入闭包:

def c={a,b,c->a+b+c};

println c([1,2,3]);

闭包可以通过将第一个或前几个参数固化为常量,使用curry方法进行拷贝:

def c={a,b,c->a+b+c};

def d=c.curry(1);//拷贝闭包c,并将第一个参数a固化为常量1

println d(2,3);//调用闭包d,只需要给出剩余的参数。实际使用的参数仍然是1,2,3

在闭包中,可以结合使用curry和不定长度参数:

def c={a,Object[] b->b.each({m->a=a+m;});a;};

def d=c.curry(1);

def e=d.curry(2,3);//实际参数为1,[2,3]

def f=e.curry(4);//实际参数为1,[2,3,4]

println f(5);//实际参数为1,[2,3,4,5]

闭包可以递归调用

三、函数

首先,必须使用def关键字定义函数,函数也不能嵌套使用;

其次,函数不能访问在外部通过def定义的变量:

def c=10;

def f(){
println c;//编译通过
}

//f();//运行时报错

还可以使用特殊的语法标记&将函数赋予新的命名:

def f(){
println "Hello,world!";
}

def s=this.&f;

s();

函数可以后参数,包括输入参数和输出参数;

如果两个函数的拥有不同个数的未指定类型的参数,则可以使用相同的名称

def f(a){
println a*a;
}

def f(a,b){
println a*b;
}

f(10);

f(5,10)

函数可以有返回值:
1.如果有显示地使用return关键字,则返回return指定的返回值,其后面的语句不再执行;
2.如果没有显式地使用return关键字,则返回函数最后一行语句的运行结果;
3.如果使用void关键字代替def关键字定义函数,则函数的返回值将为null。

如果一个函数与一个闭包拥有相同的名称和参数,则调用的时候将执行函数,而不是闭包:

def f(a){
return a*10;
}

def f={a->a*20};

println f(5);//输出50而不是100

函数可以接受闭包作为参数:

def f(a,Closure c){
c(a);
}

f(10){println "Parameter a = "+it;}

与闭包相同,函数可以为参数指定默认值:

def f(a,b=10){
return a+b;
}

println f(5);//输出15

函数也会将命名的参数包装进一个Map,并赋予第一个参数:

def f(a,b){
return b+a.x+a.y;
}

println f(x:2,1,y:3);//输出6

函数与闭包的相同之处还有,函数同样支持不定长度的参数、分解list作为参数,以及递归调用。

热点排行