一、接口默认方法
什么是默认方法
接口中的方法可以有默认实现,不需要子类去重写,在方法名前面加default关键字标识
为什么要有默认方法
Ø 为了解决接口的修改与现有的实现不兼容的问题(如JDK8集合新增了forEach方法,是在最顶层接口Iterator中将此方法标注为default,这样所有实现Iterator接口的类都可使用forEach,且子类无需再重写此方法)
Ø 例:集合类的接口Iterable
public interface Iterable<T> {
Iterator<T> iterator();
default void forEach(Consumer<? superT> action) {
Objects.requireNonNull(action);
for(T t : this) {
action.accept(t);
}
}
...
}
二、lambda表达式
首先要了解一个概念:函数式接口
Ø Jdk8中引入函数式接口的概念,即只定义了一个抽象方法的接口,上边介绍的默认方法和static方法除外(JDK8接口中可以定义static方法),可作为方法的参数,如java.lang.Runnable,java.util.Comparator等
Jdk8提供的常用函数式接口使用见附录1。
Ø 在接口声明上用注解@FunctionalInterface标注(此注解可以省略,但使用此注解时,如果接口中定义了多个方法,编译不通过,可防止其他人员在函数式接口中增加方法)
Ø 例1:函数式接口声明
@FunctionalInterface
public interface ITestA {
void print(); //只能有一个抽象方法
default void add(){ //可以有多个默认方法
System.out.println("add method");
}
}
什么是Lambda表达式
从数学和计算的角度来看,一个lambda表达式就是一个函数:对于部分或者全部输入值的组合,它会产生出一个特定的输出。在java语言中Lambda表达式引入了函数式编程的思想(函数式编程的介绍见http://www.ruanyifeng.com/blog/2012/04/functional_programming.html)。按照java传统术语来解释看,Lambda可以被理解为一种有更复杂语法的匿名方法,可以忽略修饰符,返回类型,在某些情况下参数类型同样可以省略。
一个lambda表达式的目标类型必须是一个函数式接口,并能与目标类型兼容,即lambda表达式必须具有与函数式接口定义的方法相同的参数和返回值(JRE会自动匹配lambda表达式和函数式接口)
Lambda语法
包含三个部分:
1、括号括起来的参数,对应于接口方法的参数
2、一个箭头符号:->
3、方法体,可以是表达式或代码块
例1:自定义函数式接口
@FunctionalInterface
public interface ITestA{
void add(int a,int b);
}
以下为使用lambda表达式指定函数式接口方法的具体实现
ITestA test = (a, b) -> System.out.println(a + b);
test.add(1, 3);
例2:函数式接口Runnable的使用方式
public static void sort() {
//JDK8版本之前排序方式
List<String>names = Arrays.asList("peter", "anna", "mike", "xenia");
Collections.sort(names, newComparator<String>() {
@Override
public int compare(String a, String b) {
return b.compareTo(a);
}
});
//使用Lambda表达式
Collections.sort(names, (a,b)->a.compareTo(b));
System.out.println(names);
}
例3:Lambda表达式代码块
public interface IInner {
int add(int a, int b);
}
IInner in = (a, b) -> {
int i = 2;
return a + b + i;
};
System.out.println(in.add(1, 2));
Lambda表达式作用域
Lambda表达式访问变量的作用域和匿名函数类似,只是在访问局部变量的时候,局部变量不再要求必须声明为final,但在表达式中和表达式外均不可修改局部变量的值,如例1。
同时,this作用域在匿名函数中代表匿名函数本身,而在Lambda中代表类实例,如例3
Ø 例1:访问局部变量:只能读取局部变量,不能修改
public static void localVar() {
int i = 10;
Person per = new Person("","");
List<String> sl = new ArrayList<String>();
Consumer<Integer> consumer = (s) -> {
//原则:对值封闭,对变量开放
per.setFirstName("first"); //编译通过
sl.add("a"); //编译通过
int doubleI = i * 2;
// i = 4; //此处会报编译错误
};
// i = 4; //在表达式外修改同样报编译错误
}
Ø 例2:访问成员变量:对成员变量可读可写
public class Lambda {
private static int a = 10;
private int b;
public void localVar() {
Consumer<Integer>consumer = (s) -> {
a = 5;
b = 3;
};
}
Ø 例3:this变量作用域
packagecom.jdk8.test;
public class Lambda {
public void testThis() {
List<String> names = Arrays.asList("peter", "anna", "mike", "xenia");
System.out.println(this);
Collections.sort(names, (a,b) -> {
System.out.println(this);//com.jdk8.test.Lambda@15db9742
return a.compareTo(b);
});
}
}
JDK8为什么引入lambda表达式
Ø 函数式编程(一切皆函数)可能是未来编程语言的一个趋势,lambda表达式是函数式编程的一个重要特征。同时从Java面向对象(一切皆对象)特性考虑,那么函数也是对象,可以当做参数传递
Ø 解决匿名类的一些缺点,如语法冗余、this变量易使人产生误解等(尤其是JDK8中对集合的操作增加并行处理的方式,可通过forEach方法传给集合一个处理函数,集合内部自己实现并行处理,如果使用传统的匿名内部类来实现接口的话,代码很笨重)
例1:使用Lambda表达式访问集合
public static void collectionIndex(){
List<String>sl = new ArrayList<String>();
sl.add("a");
sl.add("b");
sl.add("c");
// 不使用lambda表达式
sl.forEach(new Consumer<String>() {
@Override
public void accept(String t){
System.out.println(t);
}
});
//使用lambda表达式
Consumer<String>consumer = (s) -> System.out.println(s);
sl.forEach(consumer);
//甚至还可以这样写
sl.forEach((s) -> System.out.println(s));
}
三、方法引用
是lambda表达式的一个简化写法,所引用的方法其实是lambda表达式的方法体实现部分,如果lambda表达式方法体很长,可使用方法引用增强可读性。
语法也很简单,左边是类名或实例名,中间是"::",右边是相应的方法名。
Ø 一般方法的引用格式是
如果是静态方法,则是ClassName::methodName
如果是实例方法,则是Instance::methodName
Ø 构造函数则是ClassName::new
例1、一般方法引用
public class MethodReference {
public static void main(String[] args) {
// lambda表达式
ITestA test1 = (a, b) -> System.out.println(a + b);
test1.add(1, 2);
// 静态方法
ITestA test2 = MethodReference::anotherLongMethod;
test2.add(1, 2);
// 实例方法
MethodReferencerefrence = newMethodReference();
ITestA test3 = refrence::longMethod;
test3.add(1, 2);
}
public void longMethod(int a, int b) {
// 这是一个很长的方法体
System.out.println(a + b);
}
public static void anotherLongMethod(int a, int b) {
// 这是一个很长的方法体
System.out.println(a + b);
}
}
例2、构造函数引用
public class Person {
String firstName;
String lastName;
Person() {
}
Person(String firstName, String lastName) {
this.firstName = firstName;
this.lastName = lastName;
}
public String toString() {
return firstName + lastName;
}
public static void main(String[] args) {
IPersonFactoryfactory = Person::new;
Person per = factory.create("nick", "laus");
System.out.println(per);
}
}
public interface IPersonFactory{
Personcreate(String firstName, String lastName);
}
附录1:JDK提供的常用函数式接口
Ø Predicate<T>接口
Predicate 接口只有一个参数,返回boolean类型。该接口包含多种默认方法来将Predicate组合成其他复杂的逻辑(比如:与,或,非)
public static void predicate() {
Predicate<String> pre1 = (s) -> s.length() > 0;
Predicate<String> pre2 = (s) -> s.length() < 4;
Predicate<String> preCombined = pre1.and(pre2);
System.out.println(pre1.test("astring")); //true
System.out.println(pre1.negate().test("astring")); //false
System.out.println(preCombined.test("astring")); //false
}
Ø Function<T, R>:接收T对象,返回R对象
public static void function() {
Function<String, String> fun1 = (s) -> s + "12";
Function<String, Long> fun2 = (s) -> Long.parseLong(s);
Function<String, Long> combined = fun1.andThen(fun2);
System.out.println(combined.apply("3"));
}
Ø Supplier<T>:提供T对象(例如工厂),不接收参数
public static void supplier(){
Supplier<Person> personSupplier = Person::new;
personSupplier.get(); // new Person
}
Ø Consumer<T>:Consumer 接口表示执行在单个参数上的操作
public static void consumer(){
Consumer<Person> greeter = (p) -> System.out.println("Hello," + p.firstName);
greeter.accept(new Person("Luke", "Skywalker"));
}
Ø Comparator <T>:Java8在此之上添加了多种默认方法
public static void comparator() {
Comparator<Person> comparator = (p1, p2) -> p1.firstName
.compareTo(p2.firstName);
Person p1 = new Person("John", "Doe");
Person p2 = new Person("Alice", "Wonderland");
comparator.compare(p1, p2); // > 0
comparator.reversed().compare(p1, p2); // < 0
}
Ø 还有其他一些函数式接口可自行google
附录2:Stream接口及集合一些新操作
Stream 接口
java.util.Stream 表示能应用在一组元素上一次执行的操作序列。Stream 操作分为中间操作或者最终操作两种,最终操作返回一特定类型的计算结果,而中间操作返回Stream本身,这样你就可以将多个操作依次串起来。Stream 的创建需要指定一个数据源,比如java.util.Collection的子类,List或者Set, Map不支持。Stream的操作可以串行执行或者并行执行。
首先看看Stream是怎么用,首先创建实例代码的用到的数据List:
复制代码代码如下:
List<String> stringCollection = new ArrayList<>();
stringCollection.add("ddd2");
stringCollection.add("aaa2");
stringCollection.add("bbb1");
stringCollection.add("aaa1");
stringCollection.add("bbb3");
stringCollection.add("ccc");
stringCollection.add("bbb2");
stringCollection.add("ddd1");
Java 8扩展了集合类,可以通过 Collection.stream() 或者Collection.parallelStream() 来创建一个Stream。下面几节将详细解释常用的Stream操作:
Filter 过滤
过滤通过一个predicate接口来过滤并只保留符合条件的元素,该操作属于中间操作,所以我们可以在过滤后的结果来应用其他Stream操作(比如forEach)。forEach需要一个函数来对过滤后的元素依次执行。forEach是一个最终操作,所以我们不能在forEach之后来执行其他Stream操作。
复制代码代码如下:
stringCollection
.stream()
.filter((s) -> s.startsWith("a"))
.forEach(System.out::println);
// "aaa2", "aaa1"
Sort 排序
排序是一个中间操作,返回的是排序好后的Stream。如果你不指定一个自定义的Comparator则会使用默认排序。
复制代码代码如下:
stringCollection
.stream()
.sorted()
.filter((s) -> s.startsWith("a"))
.forEach(System.out::println);
// "aaa1", "aaa2"
需要注意的是,排序只创建了一个排列好后的Stream,而不会影响原有的数据源,排序之后原数据stringCollection是不会被修改的:
复制代码代码如下:
System.out.println(stringCollection);
// ddd2, aaa2, bbb1, aaa1, bbb3, ccc, bbb2, ddd1
Map 映射
中间操作map会将元素根据指定的Function接口来依次将元素转成另外的对象,下面的示例展示了将字符串转换为大写字符串。你也可以通过map来讲对象转换成其他类型,map返回的Stream类型是根据你map传递进去的函数的返回值决定的。
复制代码代码如下:
stringCollection
.stream()
.map(String::toUpperCase)
.sorted((a, b) -> b.compareTo(a))
.forEach(System.out::println);
// "DDD2", "DDD1","CCC", "BBB3", "BBB2", "AAA2","AAA1"
Match 匹配
Stream提供了多种匹配操作,允许检测指定的Predicate是否匹配整个Stream。所有的匹配操作都是最终操作,并返回一个boolean类型的值。
复制代码代码如下:
boolean anyStartsWithA =
stringCollection
.stream()
.anyMatch((s) ->s.startsWith("a"));
System.out.println(anyStartsWithA); // true
boolean allStartsWithA =
stringCollection
.stream()
.allMatch((s) ->s.startsWith("a"));
System.out.println(allStartsWithA); // false
boolean noneStartsWithZ =
stringCollection
.stream()
.noneMatch((s) ->s.startsWith("z"));
System.out.println(noneStartsWithZ); // true
Count 计数
计数是一个最终操作,返回Stream中元素的个数,返回值类型是long。
复制代码代码如下:
long startsWithB =
stringCollection
.stream()
.filter((s) -> s.startsWith("b"))
.count();
System.out.println(startsWithB); // 3
Reduce 规约
这是一个最终操作,允许通过指定的函数来讲stream中的多个元素规约为一个元素,规越后的结果是通过Optional接口表示的:
复制代码代码如下:
Optional<String> reduced =
stringCollection
.stream()
.sorted()
.reduce((s1, s2) -> s1 +"#" + s2);
reduced.ifPresent(System.out::println);
// "aaa1#aaa2#bbb1#bbb2#bbb3#ccc#ddd1#ddd2"
并行Streams
前面提到过Stream有串行和并行两种,串行Stream上的操作是在一个线程中依次完成,而并行Stream则是在多个线程上同时执行。
下面的例子展示了是如何通过并行Stream来提升性能:
首先我们创建一个没有重复元素的大表:
复制代码代码如下:
int max = 1000000;
List<String> values = new ArrayList<>(max);
for (int i = 0; i < max; i++) {
UUID uuid = UUID.randomUUID();
values.add(uuid.toString());
}
然后我们计算一下排序这个Stream要耗时多久,
串行排序:
复制代码代码如下:
long t0 = System.nanoTime();
long count = values.stream().sorted().count();
System.out.println(count);
long t1 = System.nanoTime();
long millis =TimeUnit.NANOSECONDS.toMillis(t1 - t0);
System.out.println(String.format("sequential sort took: %d ms",millis));
// 串行耗时: 899 ms
并行排序:
复制代码代码如下:
long t0 = System.nanoTime();
long count =values.parallelStream().sorted().count();
System.out.println(count);
long t1 = System.nanoTime();
long millis =TimeUnit.NANOSECONDS.toMillis(t1 - t0);
System.out.println(String.format("parallel sort took: %d ms",millis));
// 并行排序耗时: 472 ms
上面两个代码几乎是一样的,但是并行版的快了50%之多,唯一需要做的改动就是将stream()改为parallelStream()。
Map
前面提到过,Map类型不支持stream,不过Map提供了一些新的有用的方法来处理一些日常任务。
复制代码代码如下:
Map<Integer, String> map = new HashMap<>();
for (int i = 0; i < 10; i++) {
map.putIfAbsent(i, "val" + i);
}
map.forEach((id, val) ->System.out.println(val));
以上代码很容易理解, putIfAbsent 不需要我们做额外的存在性检查,而forEach则接收一个Consumer接口来对map里的每一个键值对进行操作。
下面的例子展示了map上的其他有用的函数:
复制代码代码如下:
map.computeIfPresent(3, (num, val) -> val + num);
map.get(3); // val33
map.computeIfPresent(9, (num, val) ->null);
map.containsKey(9); // false
map.computeIfAbsent(23, num ->"val" + num);
map.containsKey(23); // true
map.computeIfAbsent(3, num ->"bam");
map.get(3); // val33
接下来展示如何在Map里删除一个键值全都匹配的项:
复制代码代码如下:
map.remove(3, "val3");
map.get(3); // val33
map.remove(3, "val33");
map.get(3); // null
另外一个有用的方法:
复制代码代码如下:
map.getOrDefault(42, "not found"); // not found
对Map的元素做合并也变得很容易了:
复制代码代码如下:
map.merge(9, "val9", (value, newValue) -> value.concat(newValue));
map.get(9); // val9
map.merge(9, "concat", (value,newValue) -> value.concat(newValue));
map.get(9); // val9concat
Merge做的事情是如果键名不存在则插入,否则则对原键对应的值做合并操作并重新插入到map中。