lambda


JAVA8新特性之Lambda

一、Lambda

1、定义

lambda表达式是java8新增的主要特性之一,lambda表达式又称闭包或匿名函数,主要优点在于简化代码、增强代码可读性、并行操作集合等。

lambda表达式的特性:

  1. 可选类型声明: 无需声明参数类型,编译器即可自动识别
  2. 可选的参数圆括号: 仅有一个参数时圆括号可以省略
  3. 可选的大括号:主体只包含一个语句时可省略大括号
  4. 可选的返回关键字:主体只包含一个表达式返回值并省略大括号时,编译器会自动return返回值;有大括号时,需要显式指定表达式return了一个数值

特性示例:

//1、无参数,返回值1
() -> 1 
//2、无参数,无返回值
() -> System.out.print("Java8 lambda.");
//3、1个参数,参数类型为数字,返回值为其值的5倍
x ->  5 * x 
//4、2个参数,参数类型均为数字,返回值为其差值
(x, y) -> x - y
//5、2个参数,指定参数类型均为int型,返回值为其差值 
(int x, int y) -> x - y  
//6、1个参数,指定参数类型为String ,无返回值
(String str) -> System.out.print(str)

2、实际使用

1、java Runnable接口的lambda实现

用lambda表达式代替匿名类的关键在于,匿名类实现的接口使用了java.lang.FunctionalInterface注解,且只有一个待实现的抽象接口方法,如Runnable接口:

@FunctionalInterface
public interface Runnable {
    public abstract void run();
}
#java1.8之前
   new Thread(new Runnable()
   {
    @Override
    public void run()
    {
         System.out.println("No use lambda.");
    }
   }).start();
#java1.8之后
   new Thread(() -> System.out.println("Use lambda")).start();

lambda与匿名类的联系和区别
联系:
1) 都可以访问final或effectively final局部变量。
2) 生成的对象都可以调用实现的接口方法。
区别:
1) this指针的指向不同。我们知道匿名类的this指针指向匿名类,而lambda表达式的this指针指向的是包围lambda表达式的类。
2) 编译方式不同。lambda在编译器内部被翻译为私有方法,并使用了Java 7的 invokedynamic 字节码指令来动态绑定这个方法
3) 实现的接口限制有区别。匿名类可以为任意接口创建实例,只要实现接口所有的抽象方法即可;而lambda表达式只能实现函数式接口(只有一个必须实现的抽象方法)。
4) 接口默认方法的调用权限不同。匿名类实现的抽象方法允许调用接口中的默认方法,而lambda表达式不能调用接口中的默认方法。

2、java List迭代的lambda实现

java8中,集合类的顶层接口java.lang.Iterable定义了一个forEach方法:

/* @param action The action to be performed for each element
 * @throws NullPointerException if the specified action is null
 * @since 1.8
 */
default void forEach(Consumer<? super T> action) {
    Objects.requireNonNull(action);
    for (T t : this) {
        action.accept(t);
    }
}

forEach方法可以迭代集合的所有对象,其参数为Consumer对象,Consumer类位于java.util.function包下,我们看下其定义:

@FunctionalInterface//这个注解就代表函数式接口,有唯一一个带实现的抽象方法,因此可以用Lambda
public interface Consumer<T> {
    void accept(T t);

    default Consumer<T> andThen(Consumer<? super T> after) {
        Objects.requireNonNull(after);
        return (T t) -> { accept(t); after.accept(t); };
    }
}
#java1.8之前
    List<Integer> features = Arrays.asList(1,2);
    for (Integer feature : features) {
         System.out.println(feature);
     }
#java1.8之后
    List<Integer> features = Arrays.asList(1,2);
    features.forEach(n -> System.out.println(n));
#也可以——用java8的方法引用来表示(方法引用也是java8的新特性,由::操作符标示):
    List<Integer> features = Arrays.asList(1,2);
    features.forEach(System.out::println);
#注:而如果对参数有任何修改时不能使用方法引用,如:
	features.forEach(n -> System.out.println(n+1));

二、函数是接口

函数式接口(Functional Interface)是java8新增的特性,它是一个有且仅有一个抽象方法,但是可以有多个非抽象方法的接口;

用lambda表达式代替匿名类的关键在于,匿名类实现的接口使用了java.lang.FunctionalInterface注解,且只有一个待实现的抽象接口方法”, 这里的接口便是函数式接口;

Runnable接口是在JDK1.8之前已经存在的接口,在JDK1.8中加入了@FunctionalInterface注解,表示将其定义为一个函数式接口。在JDK1.8中定义的函数式接口还有:

  • java.util.concurrent.Callable
  • java.security.PrivilegedAction
  • java.util.Comparator
  • java.io.FileFilter
  • java.nio.file.PathMatcher
  • java.lang.reflect.InvocationHandler
  • java.beans.PropertyChangeListener
  • java.awt.event.ActionListener
  • javax.swing.event.ChangeListener
  • java.util.function包下的接口

文章作者: LJH
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 LJH !
  目录