一文搞懂Java8 Lambda表示式(附帶視訊教程)

java鋒哥發表於2020-08-16

Lambda表示式介紹

Java 8的一個大亮點是引入Lambda表示式,使用它設計的程式碼會更加簡潔。通過Lambda表示式,可以替代我們以前經常寫的匿名內部類來實現介面。Lambda表示式本質是一個匿名函式;

 

體驗Lambda表示式

我們通過一個小例子來體驗下Lambda表示式;

我們定義一個計算介面 只有一個方法add;

public class Program {
 
    public static void main(String[] args) {
        Cal c1=new Cal() {
            @Override
            public int add(int a, int b) {
                return a+b;
            }
        };
       int c=c1.add(1,2);
        System.out.println(c);
    }
}
 
interface Cal{
     int add(int a,int b);
}

 

這個是我們以前的實現,匿名內部類,然後呼叫執行;

我們現在用Lambda表示式改寫下:

public class Program {
 
    public static void main(String[] args) {
        Cal c1=(int a,int b) ->{return a+b;};
        int c=c1.add(1,2);
        System.out.println(c);
    }
 
    int add(int a,int b){
        return a+b;
    }
}
 
interface Cal{
     int add(int a,int b);
}

 

匿名內部類,直接改成了: 

Cal c1=(int a,int b) ->{return a+b;};

簡潔多了;

是不是感覺Lambda表示式挺強大,

接下來我們來看看Lambda表示式的語法吧;

 

Lambda表示式語法

我們看下這個Lambda表示式:

(int a,int b) ->{return a+b;};

這個本質是一個函式;

 

一般的函式類似如下:

int add(int a,int b){
  return a+b;
}

 

有返回值,方法名,引數列表,方法體 

Lambda表示式函式的話,只有引數列表,和方法體

( 引數列表 ) -> { 方法體 }

說明:

( ) :用來描述引數列表;

{ } : 用來描述方法體;

->  :Lambda運算子,可以叫做箭頭符號,或者goes to

 

Lambda表示式語法細講

我們搞一個案例,介面方法引數,無參,單個引數,兩個引數,有返回值,沒有返回值,這六種情況都羅列下:

interface If1{
 
    /**
     * 無引數無返回值
     */
     void test();
}
 
 
interface If2{
 
    /**
     * 單個引數無返回值
     * @param a
     */
    void test(int a);
}
 
interface If3{
 
    /**
     * 兩個引數無返回值
     * @param a
     * @param b
     */
    void test(int a,int b);
}
 
 
interface If4{
 
    /**
     * 無引數有返回值
     * @return
     */
    int test();
}
 
interface If5{
 
    /**
     * 單個引數有返回值
     * @param a
     * @return
     */
    int test(int a);
}
 
interface If6{
 
    /**
     * 多個引數有返回值
     * @param a
     * @param b
     * @return
     */
    int test(int a,int b);
}

 

我們用Lambda表示式實現: 

// 無引數無返回值
If1 if1=()->{
  System.out.println("無引數無返回值");
};
if1.test();
 
// 單個引數無返回值
If2 if2=(int a)->{
  System.out.println("單個引數無返回值 a="+a);
};
if2.test(3);
 
// 兩個引數無返回值
If3 if3=(int a,int b)->{
  System.out.println("兩個引數無返回值 a+b="+(a+b));
};
if3.test(2,3);
 
// 無引數有返回值
If4 if4=()->{
  System.out.print("無引數有返回值 ");
  return 100;
};
System.out.println(if4.test());
 
 
// 單個引數有返回值
If5 if5=(int a)->{
  System.out.print("單個引數有返回值 ");
  return a;
};
System.out.println(if5.test(200));
 
// 多個引數有返回值
If6 if6=(int a,int b)->{
  System.out.print("多個引數有返回值 ");
  return a+b;
};
System.out.println(if6.test(1,2));

 

執行輸出: 

無引數無返回值
單個引數無返回值 a=3
兩個引數無返回值 a+b=5
無引數有返回值 100
單個引數有返回值 200
多個引數有返回值 3

 

Lambda表示式精簡語法 

那件語法注意點:

  • 引數型別可以省略
  • 假如只有一個引數,()括號可以省略
  • 如果方法體只有一條語句,{}大括號可以省略
  • 如果方法體中唯一的語句是return返回語句,那省略大括號的同事return也要省略

 

改寫例項:

/**
 * @author java1234_小鋒
 * @site www.java1234.com
 * @company Java知識分享網
 * @create 2020-08-12 16:43
 */
public class Program2 {
 
    public static void main(String[] args) {
        // 1,引數型別可以省略
        // 2,假如只有一個引數,()括號可以省略
        // 3,如果方法體只有一條語句,{}大括號可以省略
        // 4,如果方法體中唯一的語句是return返回語句,那省略大括號的同事return也要省略
 
        // 無引數無返回值
        If1 if1=()->System.out.println("無引數無返回值");
        if1.test();
 
        // 單個引數無返回值
        If2 if2=a->System.out.println("單個引數無返回值 a="+a);
        if2.test(3);
 
        // 兩個引數無返回值
        If3 if3=(a,b)->{
            System.out.println("兩個引數無返回值 a+b="+(a+b));
        };
        if3.test(2,3);
 
        // 無引數有返回值
        If4 if4=()->100;
        System.out.println(if4.test());
 
 
        // 單個引數有返回值
        If5 if5=a->{
            System.out.print("單個引數有返回值 ");
            return a;
        };
        System.out.println(if5.test(200));
 
        // 多個引數有返回值 引數型別可以省略
        If6 if6=(a,b)->a+b;
        System.out.println(if6.test(1,2));
 
    }
 
}

 

方法引用 

有時候多個lambda表示式實現函式是一樣的話,我們可以封裝成通用方法,以便於維護;

這時候可以用方法引用實現:

語法是:物件::方法

假如是static方法,可以直接 類名::方法

 

例項如下:

public class Program2 {
 
    public static void main(String[] args) {
        // 方法引用
        // 語法:
        // static方法 類名::方法名
        // 普通方法 物件名::方法名
        Program2 program2=new Program2();
        If5 if5=program2::test;
        If5 if52=Program2::test2;
        System.out.println(if5.test(1));
        System.out.println(if52.test(1));
 
 
    }
 
    public int test(int a){
        return a-2;
    }
 
    public static int test2(int a){
        return a-2;
    }
 
}

 

構造方法引用

如果函式式介面的實現恰好可以通過呼叫一個類的構造方法來實現,

那麼就可以使用構造方法引用;

語法:類名::new

例項:

先定義一個Dog實體,實現無參和有參構造方法;

public class Dog {
 
    private String name;
 
    private int age;
 
    public Dog() {
        System.out.println("無參構造方法");
    }
 
    public Dog(String name, int age) {
        System.out.println("有參構造方法");
        this.name = name;
        this.age = age;
    }
 
 
    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;
    }
 
    @Override
    public String toString() {
        return "Dog{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

 

在定義兩個介面: 

interface DogService{
    Dog getDog();
 
}
 
interface DogService2{
    Dog getDog(String name,int age);
}

 

測試: 

public class Program3 {
 
    public static void main(String[] args) {
 
        // 普通方式
        DogService dogService=()->{
            return new Dog();
        };
        dogService.getDog();
 
        // 簡化方式
        DogService dogService2=()->new Dog();
        dogService2.getDog();
 
        // 構造方法引用
        DogService dogService3=Dog::new;
        dogService3.getDog();
 
        // 構造方法引用 有參
        DogService2 dogService21=Dog::new;
        dogService21.getDog("小米",11);
    }
}

 

執行結果: 

無參構造方法
無參構造方法
無參構造方法
有參構造方法

 

綜合例項

下面我們通過一個lambda操作集合的綜合例項,來深入體驗下Lambda表示式用法;

public class Program4 {
 
    public static void main(String[] args) {
        List<Dog> list=new ArrayList<>();
        list.add(new Dog("aa",1));
        list.add(new Dog("bb",4));
        list.add(new Dog("cc",3));
        list.add(new Dog("dd",2));
        list.add(new Dog("ee",5));
        // 排序
        System.out.println("lambda集合排序");
        list.sort((o1,o2)->o1.getAge()-o2.getAge());
        System.out.println(list);
 
        // 遍歷集合
        System.out.println("lambda遍歷集合");
        list.forEach(System.out::println);
    }
}

 

執行輸出: 

lambda集合排序
[Dog{name='aa', age=1}, Dog{name='dd', age=2}, Dog{name='cc', age=3}, Dog{name='bb', age=4}, Dog{name='ee', age=5}]
lambda遍歷集合
Dog{name='aa', age=1}
Dog{name='dd', age=2}
Dog{name='cc', age=3}
Dog{name='bb', age=4}
Dog{name='ee', age=5}

 

我們來分析下集合的sort方法,

1.jpg

sort方法裡有一個Comparator介面,再點進去看下:

2.jpg

我們通過lambda就可以輕鬆實現排序:

(o1,o2)->o1.getAge()-o2.getAge()

 

再看下集合的forEach方法,點進去:

3.jpg

有個消費者Consumer介面,再點進去:

4.jpg

介面裡有個介面引數的accept的方法;

所以我們直接方法引用 直接輸出每次的遍歷值即可;

System.out::println

 

@FunctionalInterface註解

前面我們會發現Consumer介面,Comparator介面都有

@FunctionalInterface註解;

這個註解是函式式介面註解,所謂的函式式介面,當然首先是一個介面,然後就是在這個介面裡面只能有一個抽象方法。

這種型別的介面也稱為SAM介面,即Single Abstract Method interfaces

 

特點

  • 介面有且僅有一個抽象方法

  • 允許定義靜態方法

  • 允許定義預設方法

  • 允許java.lang.Object中的public方法

 

該註解不是必須的,如果一個介面符合"函式式介面"定義,那麼加不加該註解都沒有影響。加上該註解能夠更好地讓編譯器進行檢查。如果編寫的不是函式式介面,但是加上了@FunctionInterface,那麼編譯器會報錯

 

例項

// 正確的函式式介面
@FunctionalInterface
public interface TestInterface {
  
    // 抽象方法
    public void sub();
  
    // java.lang.Object中的public方法
    public boolean equals(Object var1);
  
    // 預設方法
    public default void defaultMethod(){
     
    }
  
    // 靜態方法
    public static void staticMethod(){
  
    }
}
 
// 錯誤的函式式介面(有多個抽象方法)
@FunctionalInterface
public interface TestInterface2 {
 
    void add();
     
    void sub();
}

 

系統內建函式式介面 

Java8的推出,是以Lambda重要特性,一起推出的,其中系統內建了一系列函式式介面;

再jdk的java.util.function包下,有一系列的內建函式式介面:

5.jpg

比如常用的Consumer,Comparator,Predicate,Supplier等;

 

Lambda表示式視訊教程

感謝各位兄弟姐妹關注,鋒哥為了大夥能更深刻的掌握Lambda的原理和應用,專門錄製了一期視訊教程。主要以IDEA開發工具,來講解lambda表示式,希望小夥伴們快速的掌握。

紙上得來終覺淺,絕知此事要躬行。

需要多實戰,多思考

B站視訊教程線上地址:
https://www.bilibili.com/video/bv1ci4y1g7qD

 

------------------------------------------------------------------------------------------------------------------------------

作者: java1234_小鋒

出處:https://www.cnblogs.com/java688/p/13511720.html

版權:本站使用「CC BY 4.0」創作共享協議,轉載請在文章明顯位置註明作者及出處。

------------------------------------------------------------------------------------------------------------------------------

相關文章