Java中Lambda表達式用法介紹

Lambda

lambda是一個匿名函數,我們可以把lambda表達式理解為是一段可以傳遞的代碼。

  • lambda簡明的地將代碼或方法作為參數傳遞進去執行。
  • “函數式編程”其核心是把函數作為值。
  • 函數式接口 :隻有一個 抽象方法的接口 稱之為 函數式接口。函數式接口可以使用@FunctionalInterface進行註解。

lambda表達式拆分為兩部分

左側:lambda 表達式的參數列表

右側:lambda 表達式中所需要執行的功能,即lambda體

語法格式一:無參數,無返回值

@Test
public void test(){
    // () -> System.out.println("Hello");
    Runnable a = new Runnable(){
     @Override
     public void run(){
        System.out.println("Hello")
    }
    };
    //等同於
    Runnable a1 = () -> System.out.println("Hello");
    a1.run();
}

語法格式二:有一個參數,無返回值(若隻有一個參數 小括號可以省略不寫)

@Test
public void test(){
    //Consumer被註解@FunctionalInterface的接口(函數式接口) 唯一抽象方法 void accept(T t);
    //左側參數 -> 右側執行體
    Consumer<String> con = (x) -> System.out.println(x);
                         // x -> System.out.println(x);
    con.accept("hahah");
}

語法格式三:有兩個以上的參數,並且lambda體中有多條語句 (若lambda體中隻有一條語句,return 和 大括號都可以省略不寫)

@Test
public void test(){
    //Comparator被註解@FunctionalInterface的接口 舉例抽象方法 int compare(T o1,T o2);
    Comparator<Integer> com = (x,y) -> {
      System.out.println("hhaha0");
      return (x < y) ? -1 : ((x == y) ? 0 : 1);
    };
    com.compare(1,2);
}

註意:lambda表達式的參數類型可以省略不寫,因為jvm編譯器可以從上下文推斷出數據類型。即“類型推斷”如果要在參數裡面寫數據類型,都要寫上。

實例

實例1:

class Employee {
    private String name;
    private int age;
    private double salary;
​
    //省略 get and set and constructor
​
}
interface MyPredicate<T> {
    boolean test(T t);
}
public class Test{
    static List<Employee> list = Arrays.asList(
            new Employee("張三",10,1),
            new Employee("裡斯",20,1),
            new Employee("王五",16,1),
            new Employee("二三",30,1)
    );
    public static List<Employee> filterEmployee(List<Employee> list,MyPredicate<Employee> mp){
        List<Employee> emps = new ArrayList<>();
        for (Employee employee : list) {
            if(mp.test(employee)){
                emps.add(employee);
            }
        }
        return emps;
 }
    @org.junit.Test
    public void test1(){
        //需要使用自定義的方法
        List<Employee> list2 = filterEmployee(list,(e) -> e.getAge() >= 15);
        list2.stream().map(Employee::getName).forEach(System.out::println);
    }
    @org.junit.Test
    public void test2(){
        //可以使用stream進行list集合的過濾  不使用自定義接口
        List<Employee> list2 = list.stream().filter((e) -> e.getAge() >= 15).collect(Collectors.toList());
        list2.stream().map(Employee::getName).forEach(System.out::println);
    }
}

實例2:

創建一個MyFun接口使用@FunctionalInterface註解,並創建一個抽象方法Integer getValue(Integer num);在Test類對變量進行某種操作。

@FunctionalInterface
interface MyFun{
    Integer getValue(Integer num);
}
public class Test{
    @org.junit.Test
    public void Test(){
        operation(100,num -> ++num);
    }
    /**
    * param1 num : 傳入的整形數
    * param2 mf : 實現某種方式對 整形數 進行操作。
    **/
    public Integer operation(Integer num,MyFun mf){
        return mf.getValue(num);
    }
}
class Employee {
    private String name;
    private int age;
    private double salary;
​
    @Override
    public String toString() {
        return "["+this.name+","+this.getAge()+","+this.getSalary()+"]";
    }
    //省略 getter and setter  and constructor
}
​
public class Test {
    List<Employee> list = Arrays.asList(
            new com.bilibili.lambda.test1.Employee("張三",10,1),
            new com.bilibili.lambda.test1.Employee("裡斯",20,1),
            new com.bilibili.lambda.test1.Employee("王五",16,1),
            new Employee("二三",30,1)
    );
    @org.junit.Test
    public void test(){
        Collections.sort(list,(e1,e2) -> {
            if(e1.getAge() == e2.getAge()){
                return e1.getName().compareTo(e2.getName());
            }else{
                //比較年齡大小
                return Integer.compare(e1.getAge(),e2.getAge());
            }
        });
        for (Employee e: list) {
            System.out.println(e);
        }
    }
}

四大核心函數式接口

  • Consumer<T> : 消費性接口 void accept(T t);
  • Supplier<T> : 共給性接口 T get();
  • Function<T,R> : 函數性接口 T代表參數,R代表返回值 R apply(T t);
  • Predicate<T> :斷言性接口 boolean test(T t);
 
class Test{
    @org.junit.Test
    publilc void test(){
        happy(10000,(money)->System.out.println("happy消費"+money+"元"));
    }
    public void happy(double money,Consumer<double> con){
        con.accept(money);
    }
}

lambda方法引用

方法引用:若lambda體中的內同有方法已經實現瞭,我們可以使用“方法引用”

(可以理解為方法引用時lambda的另一種表現形式)

主要有三種語法格式:

  • 對象::實例方法名
  • 類::靜態方法名
  • 類::實例方法名
class Test{
    //對象::實例方法名
    @org.junit.Test
    public void test(){
        Consumer<String> con = (x) -> System.out.println(x);
        con.accept("haha");
        Consumer<String> con2 = System.out::println;
        con2.accept("haha");
    }
    //類::靜態方法名
    @org.junit.Test
    public void test2(){
        Comparator<Integer> com = (x,y) -> Integer.compare(x,y);
        Comparator<Integer> com2 = Integer::compare;
        com.compare(1,2);
        com2.compare(1,2);
    }
    //類::實例方法名
    @org.junit.Test(){
        BiPredicate<String,String> bp = (x,y) -> x.equals(y);
        bp.test("a","a");
        BiPredicate<String,String> bp2 = String::equals;
    }
}

lambda構造器引用

格式:

CalssName::new

class Test{
    @org.junit.Test
    public void test(){
        Supplier<String> sup = () -> new String();
        //這裡的構造器引用取決於 接口方法的參數 的個數。 此處函數式接口 T get(); 為無參抽象方法所以String在實例化時 也是實例化無參的構造方法  其他類也適用
        Supplier<String> sup2 = String::new;
        String str = sup2.get();
    }
}

到此這篇關於Java中Lambda表達式用法介紹的文章就介紹到這瞭。希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。

推薦閱讀: