java8新特性

本文详细介绍了Java 8的三大核心特性:接口默认方法,用于解决接口修改兼容问题;Lambda表达式,引入函数式编程思想,简化匿名类使用;方法引用,作为Lambda表达式的简化形式,提高代码可读性。通过实例解析了这些新特性的用法和作用,帮助开发者更好地理解和运用Java 8。
摘要由CSDN通过智能技术生成

一、接口默认方法

什么是默认方法

接口中的方法可以有默认实现,不需要子类去重写,在方法名前面加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

附录2Stream接口及集合一些新操作

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中。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值