JDK1.8新特性1
- JDK1.8新特性:
- Lambda表达式:
- 使用:
- 无参数无返回值:
- 单参数无返回值:
- 多参数无返回值:
- 多参数有返回值:
- 案例:
- 案例1:
- 案例2:
- 案例3:
- 函数式接口:
- 四大核心函数式接口:
- 衍生接口:
- 使用:
- 使用1:
- 使用2:
- 方法、构造方法和数组引用:
- 方法引用:
- 对象::实例方法
- 类名::静态方法
- 类名::实例方法
- 构造方法引用:
- 数组引用:
JDK1.8新特性:
- 速度更快 - 优化底层源码,比如HashMap、ConcurrentHashMap。
- 代码更少 - 添加新的语法Lambda表达式。
- 强大的Stream API。
- 便于并行。
- 最大化减少空指针异常 - Optional。
Lambda表达式:
Lambda是一个匿名函数(方法), 允许把函数作为一个方法的参数 。
作用: 1. 一般都是优化匿名内部类。
2. 利用Lambda表达式可以写出更简洁、更灵活的代码。
使用:
tips:
- 重写方法的形参只有一个时,可以不加小括号。
- Lambda表达式当中不允许声明一个与局部变量同名的参数或者局部变量。
- Lambda表达式中访问外层的局部变量,外层的局部变量自动变成隐式常量,默认添加final。
- 重写方法的形参同时加类型或同时不加类型。
无参数无返回值:
当匿名内部类中只有一个方法时。
public interface I1 {public void method();
}
public class Test01 {public static void main(String[] args) {I1 i1 = new I1() {@Overridepublic void method() {System.out.println("使用传统匿名内部类的方式");}};i1.method();//当匿名内部类中只有一个方法时,才能使用lambda表达式这样简化。I1 i2 = ()->{System.out.println("使用lambda表达式的方式");};i2.method();//当匿名内部类中只有一个方法时,才能使用lambda表达式这样简化。I1 i3 = ()->System.out.println("使用lambda表达式的方式");i3.method();}
}
单参数无返回值:
当匿名内部类中只有一个方法,且该方法只有一个参数。
public interface I1 {public void method(String str);
}
public class Test01 {public static void main(String[] args) {I1 i1 = new I1() {@Overridepublic void method(String str) {System.out.println("使用传统匿名内部类的方式:" + str);}};i1.method("今天天气真好!");I1 i2 = (String str)->{System.out.println("使用lambda表达式的方式:" + str);};i2.method("今天天气真好!");I1 i3 = (String str)->System.out.println("使用lambda表达式的方式:" + str);i3.method("今天天气真好!");//省略参数的数据类型I1 i3 = (str)->System.out.println("使用lambda表达式的方式:" + str);i3.method("今天天气真好!");//省略方法的小括号//方法的形参只有一个时,可以不加小括号。I1 i4 = str->System.out.println("使用lambda表达式的方式:" + str);i4.method("今天天气真好!");}
}
多参数无返回值:
当匿名内部类中只有一个方法,且该方法只有多个参数。
public interface I1 {public void method(String str,int i);
}
public class Test01 {public static void main(String[] args) {I1 i1 = new I1() {@Overridepublic void method(String str, int i) {System.out.println("使用传统匿名内部类的方式:" + str + " -- " + i);}};i1.method("今天天气真好!", 666);I1 i2 = (String str,int i)->{System.out.println("使用lambda表达式的方式:" + str + " -- " + i);};i2.method("今天天气真好!", 777);I1 i3 = (String str,int i)->System.out.println("使用lambda表达式的方式:" + str + " -- " + i);i3.method("今天天气真好!", 888);//多个参数时,参数的数据类型要么全部去掉,要么全部保留。I1 i4 = (str, i)->System.out.println("使用lambda表达式的方式:" + str + " -- " + i);i4.method("今天天气真好!", 999);}
}
多个参数时,参数的数据类型要么全部去掉,要么全部保留。
多参数有返回值:
当匿名内部类中只有一个方法,且该方法只有多个参数,且有返回值时。
public interface I1 {public String method(int a,int b);
}
public class Test01 {public static void main(String[] args) {I1 i1 = new I1() {@Overridepublic String method(int a, int b) {return "使用传统匿名内部类的方式:" + (a+b);}};String method = i1.method(10, 10);System.out.println(method);I1 i2 = (int a,int b)->{return "使用lambda表达式的方式:" + (a+b);};String method = i2.method(20, 20);System.out.println(method);I1 i3 = (a, b)-> "使用lambda表达式的方式:" + (a+b);String method = i3.method(30, 30);System.out.println(method);}
}
案例:
案例1:
调用Collections.sort()方法,通过定制排序比较两个Student对象(先按年龄比较,年龄相同按照薪资比较),使用Lambda表达式作为参数传递。
//枚举
public enum Course {JAVA,HTML,PYTHON;
}
public class Student {private String name;private int age;private double salary;private Course course;public Student() {}public Student(String name, int age, double salary, Course course) {this.name = name;this.age = age;this.salary = salary;this.course = course;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public double getSalary() {return salary;}public void setSalary(double salary) {this.salary = salary;}@Overridepublic int hashCode() {final int prime = 31;int result = 1;result = prime * result + age;result = prime * result + ((course == null) ? 0 : course.hashCode());result = prime * result + ((name == null) ? 0 : name.hashCode());long temp;temp = Double.doubleToLongBits(salary);result = prime * result + (int) (temp ^ (temp >>> 32));return result;}@Overridepublic boolean equals(Object obj) {if (this == obj)return true;if (obj == null)return false;if (getClass() != obj.getClass())return false;Student other = (Student) obj;if (age != other.age)return false;if (course != other.course)return false;if (name == null) {if (other.name != null)return false;} else if (!name.equals(other.name))return false;if (Double.doubleToLongBits(salary) != Double.doubleToLongBits(other.salary))return false;return true;}@Overridepublic String toString() {StringBuilder builder = new StringBuilder();builder.append("Student [name=");builder.append(name);builder.append(", age=");builder.append(age);builder.append(", salary=");builder.append(salary);builder.append(", course=");builder.append(course);builder.append("]");return builder.toString();}}
public static void main(String[] args) {List<Student> stuList = Arrays.asList(new Student("张三", 28, 4800,Course.JAVA),new Student("李四", 36, 7200,Course.JAVA),new Student("王五", 19, 9600,Course.HTML),new Student("赵六", 42, 6100,Course.HTML),new Student("孙七", 23, 9600,Course.PYTHON),new Student("吴八", 28, 3000,Course.PYTHON));//匿名内部类的方式
// Collections.sort(stuList, new Comparator<Student>() {
// @Override
// public int compare(Student o1, Student o2) {
// if(o1.equals(o2)){
// return 0;
// }
//
// int compare = Integer.compare(o1.getAge(), o2.getAge());
// if(compare != 0){
// return compare;
// }
//
// return Double.compare(o1.getSalary(), o2.getSalary());
// }
// });Collections.sort(stuList, (o1,o2)->{if(o1.equals(o2)){return 0;}int compare = Integer.compare(o1.getAge(), o2.getAge());if(compare != 0){return compare;}return Double.compare(o1.getSalary(), o2.getSalary());});for (Student stu : stuList) {System.out.println(stu);}}
案例2:
创建I1接口,创建抽象方法:public String getValue(String str),在测试类中编写方法使用接口作为参数,将一个字符串转为大写,并作为方法的返回值。
public class Test01 {public static void main(String[] args) {String str = method("abc", (x)-> {return x.toUpperCase();});System.out.println(str);}public static String method(String str,I1 i1){return i1.getValue(str);}
}interface I1{public String getValue(String str);
}
案例3:
创建I1<T,R>接口,泛型T为参数,R为返回值,创建抽象方法:public R add(T t1,T t2),在测试类中编写方法使用接口作为参数,计算两个long类型的和。
public class Test01 {public static void main(String[] args) {long addLong = addLong(100,200,(a,b)->{return a+b;});System.out.println(addLong);}public static long addLong(long l1,long l2,I1<Long,Long> i1){return i1.add(l1, l2);}
}interface I1<T,R>{public R add(T t1,T t2);
}
函数式接口:
函数式接口:是指仅仅只包含一个抽象方法的接口,jdk1.8提供了一个@FunctionalInterface注解来定义函数式接口,如果我们定义的接口不符合函数式的规范便会报错。配合Lambda表达式一起使用。
四大核心函数式接口:
应用场景:当项目中需要一个接口,并且该接口中只有一个抽象方法,就没必要去创建新的接口,直接选择Java提供的使用合适的函数式接口即可。
函数式接口 | 参数类型 | 返回类型 | 用途 |
---|---|---|---|
Consumer 消费型接口 | T | void | void accept(T t); |
Supplier 供给型接口 | void | T | T get(); |
Function<T, R> 函数型接口 | T | R | R apply(T t); |
Predicate 断言型接口 | T | boolean | booelan test(T t); |
衍生接口:
函数式接口 | 参数类型 | 返回类型 | 用途 |
---|---|---|---|
BiConsumer<T, U> | T,U | void | 对类型为T,U参数应用操作。包含方法为void accept(T t,U u); |
BiFunction<T, U, R> | T,U | R | 对类型为T,U参数应用操作,并返回R类型的结果。包含方法为R apply(T t,U u); |
UnaryOperator extends Function<T, T> | T | T | 对类型为T的对象进行一元运算,并返回T类型的结果。包含方法为T apply(T t); |
BinaryOperator extends BiFunction<T,T,T> | T,T | T | 对类型为T的对象进行二元运算,并返回T类型的结果。包含方法为T apply(T t1,T t2); |
ToIntFunction ToLongFunction ToDoubleFunction | T | int long double | 分别计算int、long、double值的函数 |
IntFunction LongFunction DoubleFunction | int long double | R | 参数为int、long、double类型的函数 |
使用:
使用1:
创建I1接口,创建抽象方法:public String getValue(String str),在测试类中编写方法使用接口作为参数,将一个字符串转为大写,并作为方法的返回值。
public class Test01 {public static void main(String[] args) {String str = method("abc", (x)-> {return x.toUpperCase();});System.out.println(str);}//Function函数式接口public static String method(String str,Function<String, String> fun){//apply函数式接口的方法return fun.apply(str);}
}
再更新:
public class Test01 {public static void main(String[] args) {String str = method("abc", (x)->{return x.toUpperCase();});System.out.println(str);}//UnaryOperator函数式接口public static String method(String str,UnaryOperator<String> uo){apply函数式接口的方法return uo.apply(str);}
}
使用2:
创建I1<T,R>接口,泛型T为参数,R为返回值,创建抽象方法:public R add(T t1,T t2),在测试类中编写方法使用接口作为参数,计算两个long类型的和。
public class Test01 {public static void main(String[] args) {long addLong = addLong(100,200,(a,b)->{return a+b;});System.out.println(addLong);}public static long addLong(long l1,long l2,BiFunction<Long,Long,Long> bf){return bf.apply(l1, l2);}
}
再更新:
public class Test01 {public static void main(String[] args) {long addLong = addLong(100,200,(a,b)->{return a+b;});System.out.println(addLong);}public static long addLong(long l1,long l2,BinaryOperator<Long> bo){return bo.apply(l1, l2);}
}
方法、构造方法和数组引用:
方法、构造方法和数组引用就是Lambda的另一种表现形式,可以简化Lambda表达式。
方法引用:
若Lamdba表达式中的内容由方法已经实现了,可以使用方法引用这个技能,当你需要使用方法引用时,目标引用放在分隔符::前,方法的名称放在后面。
对象::实例方法
Lambda表达式中调用方法的参数类型和返回值必须和函数式接口中的抽象方法一致。
public class Test01 {public static void main(String[] args) {I1 i1 = new I1() {@Overridepublic void method(String str) {//println方法无返回值,一个参数,参数类型是StringSystem.out.println(str);}};i1.method("今天天气真好!");I1 i2 = (str)->System.out.println(str);i2.method("今天天气真好!");// 因为 method方法无返回值,一个参数,参数类型是String,//且println方法无返回值,一个参数,参数类型是String//故可以使用方法的引用I1 i3 = System.out::println;i3.method("今天天气真好!");}
}interface I1{//method无返回值,一个参数,参数类型是Stringpublic void method(String str);
}
因为 method 方法无返回值,一个参数,参数类型是 String ,且 println 方法无返回值,一个参数,参数类型是 String ,一 一对应,故可以使用方法的引用。
类名::静态方法
Lambda表达式中调用方法的参数类型和返回值必须和函数式接口中的抽象方法一致。
public class Test01 {public static void main(String[] args) {Comparator<Integer> comparator = new Comparator<Integer>() {@Overridepublic int compare(Integer o1, Integer o2) {return Integer.compare(o1, o2);}};int compare1 = comparator.compare(10, 20);System.out.println(compare1);Comparator<Integer> comparator = (o1,o2)-> Integer.compare(o1, o2);int compare2 = comparator.compare(10, 20);System.out.println(compare2);Comparator<Integer> comparator = Integer::compare;int compare3 = comparator.compare(10, 20);System.out.println(compare3);}
}
类名::实例方法
Lambda表达式参数列表中第一个参数必须是实例方法的调用者,第二个参数必须是实例方法的参数。
public class Test01 {public static void main(String[] args) {I1<String> i1 = new I1<String>() {@Overridepublic boolean method(String t1, String t2) {return t1.equals(t2);}};boolean method = i1.method("abc", "abc");System.out.println(method);I1<String> i2 = (str1,str2)->str1.equals(str2);boolean method2 = i2.method("abc", "abc");System.out.println(method2);//重写method方法,该方法的第一个参数调用equals,第二个参数是传入equals里的数据I1<String> i3 = String::equals;boolean method3 = i3.method("abc", "abc");System.out.println(method3);}
}interface I1<T>{public boolean method(T t1,T t2);
}
构造方法引用:
类名::new
需要调用的构造方法的参数列表必须和函数式接口中抽象方法的参数列表一致。
public class Test01 {public static void main(String[] args) {I1 i = new I1() {@Overridepublic Student method() {return new Student();}};Student st = i.method();System.out.println(st);//调用无参构造去创建学生对象//I1的method方法无参数I1 i1 = Student::new;Student stu1 = i1.method();System.out.println(stu1);//调用有参构造去创建学生对象//I2的method方法有参数I2 i2 = Student::new;Student stu2 = i2.method("小明", 23, 12000, Course.JAVA);System.out.println(stu2);}
}
interface I1{public Student method();
}
interface I2{public Student method(String name, int age, double salary, Course course);
}
数组引用:
type[]::new
public class Test01 {public static void main(String[] args) {//参数 方法里面的语句I1<String[]> i = (capacity)->new String[capacity];String[] s = i.method(3);System.out.println(Arrays.toString(s));I1<String[]> i1 = String[]::new;String[] ss = i1.method(3);System.out.println(Arrays.toString(ss));}
}interface I1<T>{public T method(int capacity);
}