Java 8新特性汇总
一、Lambda表达式
1.Lambda表达式概述
Lambda是一个匿名函数,可以把Lambda表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升
2.Lamdba表达式基本语法
3.Lambda表达式使用总结
- -> 左边:'lambda’形参列表的参数类型可以省略(类型推断);如果lambda形参列表只有一个参数,其一对()也可以省略
- -> 右边:lambda体应该使用一对{}包裹;如果lambda体只有一条执行语句(可能是return语句),省略这一对{}和return关键字
4.Lambda代码示例
public class Lambda1Test {
//语法格式一:无参,无返回值
@Test
public void test1(){
Runnable r1 = new Runnable() {
@Override
public void run() {
System.out.println("我爱北京天安门");
}
};
r1.run();
System.out.println("**********使用Lambda表达式*********");
Runnable r2 = () -> {
System.out.println("我爱北京天安门");
};
r2.run();
}
//语法格式二:Lambda需要一个参数,但是没有返回值
@Test
public void test2(){
Consumer<String> con1 = new Consumer<String>() {
@Override
public void accept(String s) {
System.out.println(s);
}
};
con1.accept("我爱北京天安门");
System.out.println("***************");
Consumer<String> con2 = (String s) -> {
System.out.println(s);
};
con2.accept("我爱北京天安门");
}
//语法格式三:数据类型可以省略,因为可由编译器推断得出,称为“类型推断”
@Test
public void test3(){
Consumer<String> con1 = new Consumer<String>() {
@Override
public void accept(String s) {
System.out.println(s);
}
};
con1.accept("我爱北京天安门");
System.out.println("***************");
Consumer<String> con2 = (s) -> {
System.out.println(s);
};
con2.accept("我爱北京天安门");
}
//其他的类型推断
@Test
public void test4(){
ArrayList<String> list = new ArrayList<>();//类型推断
int[] arr = {
1,2,3};//类型推断
}
//语法格式四:Lambda 若只需要一个参数时,参数的小括号可以省略
@Test
public void test5(){
Consumer<String> con1 = new Consumer<String>() {
@Override
public void accept(String s) {
System.out.println(s);
}
};
con1.accept("我爱北京天安门");
System.out.println("***************");
Consumer<String> con2 = s -> {
System.out.println(s);
};
con2.accept("我爱北京天安门");
}
//语法格式五:Lambda 需要两个或以上的参数,多条执行语句,并且可以有返回值
@Test
public void test6(){
Comparator<Integer> com1 = new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
System.out.println(o1);
System.out.println(o2);
return Integer.compare(o1,o2);
}
};
System.out.println(com1.compare(12,34));
System.out.println("*********");
Comparator<Integer> com2 = (o1,o2) -> {
System.out.println(o1);
System.out.println(o2);
return Integer.compare(o1,o2);
};
System.out.println(com2.compare(23,38));
}
//语法格式六:当 Lambda 体只有一条语句时,return 与大括号若有,都可以省略
@Test
public void test7(){
Comparator<Integer> com1 = new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
return Integer.compare(o1,o2);
}
};
System.out.println(com1.compare(12,34));
System.out.println("*********");
Comparator<Integer> com2 = (o1,o2) -> Integer.compare(o1,o2);
System.out.println(com2.compare(23,38));
}
}
5.Lamdba表达式总结
- Lambda表达式的本质:作为函数式接口的实例
- 如果一个接口中,只声明了一个抽象方法,则此接口就称为函数式接口。我们可以在一个接口上使用 @FunctionalInterface 注解,这样做可以检查它是否是一个函数式接口。
- 因此以前用匿名实现类表示的现在都可以用Lambda表达式来写。
二、函数式(Functional)接口(未懂,开发中可有可无)
三、方法引用、构造器引用和数组引用(未懂,基于函数式(Functional)接口)
public interface Comparator { int compare(T o1, T o2); boolean equals(Object obj); }
注意最后这个Comparator接口。它里面声明了两个方法,貌似不符合函数式接口的定义,但它的确是函数式接口。这是因为equals方法是Object的,所有的接口都会声明Object的public方法——虽然大多是隐式的。所以,Comparator显式的声明了equals不影响它依然是个函式数接口。
你可以用一个λ表达式为一个函数式接口赋值:
Runnable r1 = () -> {System.out.println(“Hello Lambda!”);};
然后再赋值给一个Object:
Object obj = r1;
但却不能这样干:
Object obj = () -> {System.out.println(“Hello Lambda!”);}; // ERROR! Object is not a functional interface!
必须显式的转型成一个函数式接口才可以:
Object o = (Runnable) () -> { System.out.println(“hi”); }; // correct
Lambda表达式是对象,而不是函数,它们必须依附于一类特别的
对象类型——函数式接口。
Lambda表达式就是一个函数式接口的实例