Java8(一)--lambda表达式

  相信作为一个Java程序员都会或多或少的了解过Java8中的lambda表达式、函数式编程等,本人也是用过lambda表达式,使用的都是比较简单

的实现

通过一个例子去都感受lambda:

Comparator<Student> comparator = new Comparator<Student>() {
    @Override
    public int compare(Student o1, Student o2) {
        return o1.getStudentAge() - o2.getStudentAge();
    }
};
Comparator<Student> comparator1 = (Student o1, Student o2) -> o1.getStudentAge() - o2.getStudentAge();

一、什么是lambda表达式?

  可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主体、返回类型,可能还有一个可以抛出的异常列表。

1、匿名:

  我们说匿名,是因为它不像普通的方法那样有一个明确的名称:写得少而想得多!

2、函数:

  我们说它是函数,是因为Lambda函数不像方法那样属于某个特定的类。但和方法一样,Lambda有参数列表、函数主体、返回类型,还可能有可以

抛出的异常列表。

3、传递:

  Lambda表达式可以作为参数传递给方法或存储在变量中。

4、简洁:

  无需像匿名类那样写很多模板代码。

上面的定义有点长,我们主要是去理解lambda的特点。lambda这个词自于学术界开发出来的一套用来描述计算的λ演算法

二、lambda表达式结构

1、参数:采用了Comparator中compare方法的参数,两个Student。

2、箭头:箭头 -> 把参数列表与Lambda主体分隔开。

3、主体:比较两个Student的年龄。表达式就是Lambda的返回值了。

基本语法

1、(parameters) -> expression

2、(parameters) -> { statements; }  //请注意语句的花括号

举个栗子

(String s) -> s.length()
(int x, int y) -> {
 System.out.println("Result:");
 System.out.println(x+y);
}

第一个例子:参数为String类型并返回一个 int 。Lambda没有 return 语句,因为已经隐含了 return,对应第一种语法

第二个例子:具有两个 int 类型的参数而没有返回值( void返回)。注意Lambda表达式可以包含多行语句,对应第二种语法

错误示例:

 () -> {return "Mario";}                  //这也是正确的lambda,显式返回String
 (Integer i) -> return "Alan" + i;      // 错误的lambda,需要使用花括号
 (String s) -> {"IronMan";}             //这是一个表达式,不是一个语句,可以改成
 (String s) -> "Iron Man"或者 (String s)->{return "IronMan";}

三、在哪里使用、怎么使用lambda

函数式接口

  就是有且只有一个抽象方法的接口,例如Runnable,Callable

  Lambda表达式允许以内联的形式为为函数式接口的抽象方法提供实现,并把整个表达式作为函数式接口的实例

举个栗子

Runnable r1 = () -> System.out.println("Hello World 1");  //使用lambda表达式
Runnable r2 = new Runnable(){                  //使用匿名内部类
    public void run(){
        System.out.println("Hello World 2");
    }
};

PS:default、static方法是不算的,照样还是函数式接口

@FunctionalInterface
public interface abc{

    void add();
    static void del() {
        System.out.println("aaa");
    };
    default void update(){
        System.out.println("bbb");
    };
}

函数描述符

  函数式接口的抽象方法的签名基本上就是Lambda表达式的签名。我们将这种抽象方法叫作函数描述符。

例如:

  1、Runnable接口只有一个run(),而且返回值是void,所以签名就是() -> void,代表参数为空,返回值为void

  2、Callable接口只有一个call(),而且返回值是T(你可以通过String或者其他数据类型去代替),签名是 () -> String

来个反面教材

execute(() -> "abc");                        //返回值定义为String,签名() -> String,就和Runnable中的抽象方法run的签名不相匹配了
public void execute(Runnable r){
    r.run();
}                

  Lambda表达式可以被赋给一个变量,或传递给一个接受函数式接口作为参数的方法,当然这个Lambda表达式的签名要和函数式接口的抽象方法一样

举个栗子

public void process(Runnable r){
    r.run();
}
process(() -> System.out.println("This is awesome!!"));

结果:

This is awesome!!

相当于

test.process(new Runnable() {
    @Override
    public void run() {
        System.out.println("This is awesome!!");
    }
});

@FunctionalInterface

  表示该接口会设计成一个函数式接口,如果不是函数式接口,注解就会报错,表示存在多个抽象方法,这个注解不是必须的,就像

@Override一样

很多接口在Java8中都添加了这个注解

@FunctionalInterface
public interface Comparator<T> {}

@FunctionalInterface
public interface Runnable {)

猜你喜欢

转载自www.cnblogs.com/huigelaile/p/10992915.html