Java8新特性之Lambda表达式浅析
说到java 8,首先会想到lambda(闭包)以及虚拟扩展方法(default method),这个特性早已经被各大技术网站炒得沸沸扬扬了,也是我们java 8系列开篇要讲的第一特性(jep126 http://openjdk.java.net/jeps/126),jdk8的一些库已经应用了lambda表达式重新设计了,理解他对学习java 8新特性有着重要的意义。
一、函数式接口
函数式接口(functional interface 也叫功能性接口,其实是同一个东西)。简单来说,函数式接口是只包含一个方法的接口。比如java标准库中的java.lang.runnable和java.util.comparator都是典型的函数式接口。java 8提供 @functionalinterface作为注解,这个注解是非必须的,只要接口符合函数式接口的标准(即只包含一个方法的接口),虚拟机会自动判断,但 最好在接口上使用注解@functionalinterface进行声明,以免团队的其他人员错误地往接口中添加新的方法。
java中的lambda无法单独出现,它需要一个函数式接口来盛放,lambda表达式方法体其实就是函数接口的实现,下面讲到语法会讲到
二、lambda语法
包含三个部分
1.一个括号内用逗号分隔的形式参数,参数是函数式接口里面方法的参数
2.一个箭头符号:->
3.方法体,可以是表达式和代码块,方法体函数式接口里面方法的实现,如果是代码块,则必须用{}来包裹起来,且需要一个return 返回值,但有个例外,若函数式接口里面方法返回值是void,则无需{}
总体看起来像这样:
(parameters) -> expression 或者 (parameters) -> { statements; }
看一个完整的例子,方便理解
/**
* 测试lambda表达式
*
* @author benhail
*/
public class testlambda {
public static void runthreaduselambda() {
//runnable是一个函数接口,只包含了有个无参数的,返回void的run方法;
//所以lambda表达式左边没有参数,右边也没有return,只是单纯的打印一句话
new thread(() ->system.out.println("lambda实现的线程")).start();
}
public static void runthreaduseinnerclass() {
//这种方式就不多讲了,以前旧版本比较常见的做法
new thread(new runnable() {
@override
public void run() {
system.out.println("内部类实现的线程");
}
}).start();
}
public static void main(string[] args) {
testlambda.runthreaduselambda();
testlambda.runthreaduseinnerclass();
}
}
可以看出,使用lambda表达式设计的代码会更加简洁,而且还可读。
三、方法引用
其实是lambda表达式的一个简化写法,所引用的方法其实是lambda表达式的方法体实现,语法也很简单,左边是容器(可以是类名,实例名),中间是"::",右边是相应的方法名。如下所示:
一般方法的引用格式是
如果是静态方法,则是classname::methodname。如 object ::equals
如果是实例方法,则是instance::methodname。如object obj=new object();obj::equals;
构造函数.则是classname::new
再来看一个完整的例子,方便理解:
import java.awt.flowlayout;
import java.awt.event.actionevent;
import javax.swing.jbutton;
import javax.swing.jframe;
/**
*
* @author benhail
*/
public class testmethodreference {
public static void main(string[] args) {
jframe frame = new jframe();
frame.setlayout(new flowlayout());
frame.setvisible(true);
jbutton button1 = new jbutton("点我!");
jbutton button2 = new jbutton("也点我!");
frame.getcontentpane().add(button1);
frame.getcontentpane().add(button2);
//这里addactionlistener方法的参数是actionlistener,是一个函数式接口
//使用lambda表达式方式
button1.addactionlistener(e -> { system.out.println("这里是lambda实现方式"); });
//使用方法引用方式
button2.addactionlistener(testmethodreference::dosomething);
}
/**
* 这里是函数式接口actionlistener的实现方法
* @param e
*/
public static void dosomething(actionevent e) {
system.out.println("这里是方法引用实现方式");
}
}
可以看出,dosomething方法就是lambda表达式的实现,这样的好处就是,如果你觉得lambda的方法体会很长,影响代码可读性,方法引用就是个解决办法
四、总结
以上就是lambda表达式语法的全部内容了,相信大家对lambda表达式都有一定的理解了,但只是代码简洁了这个好处的话,并不能打动很多观众,java 8也不会这么令人期待,其实java 8引入lambda迫切需求是因为lambda 表达式能简化集合上数据的多线程或者多核的处理,提供更快的集合处理速度 ,这个后续会讲到,关于jep126的这一特性,将分3部分,之所以分开,是因为这一特性可写的东西太多了,这部分让读者熟悉lambda表达式以及方法引用的语法和概念,第二部分则是虚拟扩展方法(default method)的内容,最后一部分则是大数据集合的处理,解开lambda表达式的最强作用的神秘面纱。敬请期待。。。。
上一篇: Java中避免空指针异常的方法
下一篇: MySQL基本命令、常用命令总结