首頁 > Java > java教程 > 主體

Java8 Lamdba函數式推導的語法怎麼表達

WBOY
發布: 2023-04-14 19:10:20
轉載
1270 人瀏覽過

前言

有且只有一個抽象函數的接口就是函數式接口,利用函數式接口我們就可以創建lamdba,但是其中可以包括靜態方法和default

1,lamdba表達式的語法

首先我們來看一個簡單的lamdba表達式的應用,就是在創建線程時候

        //创建一个线程,将线程的名字打印出来
        new Thread(()-
       System.out.println(Thread.currentThread().getName())).start();
        Thread.currentThread().join();
        //Thread-0
登入後複製

看下面的例子:

        Comparator<Apple> byColor = new Comparator<Apple>() {
            @Override
            public int compare(Apple o1, Apple o2) {
                return o1.getColor().compareTo(o2.getColor());
            }
        };
        listApple.sort(byColor);
        Comparator<Apple> byCOlor2 = (o1,o2) -> o1.getColor().compareTo(o2.getColor());
登入後複製

我們不難看到lamdba的語法構成如下:

#參數函數體就像上面的o1 和o2就是參數,後面的是函數體(Apple a) -> a.getColor() (parameters)-> expression (parameters) ->{statements}

compareTo函數介紹:

 // 根据Unicode来判断,返回一个boolean 根据boolean来说明,这两个对象需不需要交换,其实我们可以不用这个CompareTo,我们可以自己写判断条件,只要是返回boolean就行
  public int compareTo(String anotherString) {
        int len1 = value.length;
        int len2 = anotherString.value.length;
        int lim = Math.min(len1, len2);
        char v1[] = value;
        char v2[] = anotherString.value;

        int k = 0;
        while (k < lim) {
            char c1 = v1[k];
            char c2 = v2[k];
            if (c1 != c2) {
                return c1 - c2;
            }
            k++;
        }
        return len1 - len2;
    }
登入後複製

2,常見的函數式介面

  • Predicate boolean test(T t)

  • Consumer accept(T t)

  • Function R apple(T,t)

  • Supplier T get()

#Predicate :傳入一個變數回傳一個Boolean值

private static List<Apple> filterApple(List<Apple> list, Predicate<Apple> predicate){
        List<Apple> lists = new ArrayList<>();
        for (Apple apple:list){
            if (predicate.test(apple)){
                lists.add(apple);
            }
        }
        return lists;
    }
//使用
        List<Apple> list = filterApple(listApple, apple -> apple.getWeight() > 100);
        System.out.println(list);
//我们发现在 filterApple 函数里面我们传入了两个参数,一个是list 一个是Predicate抽象函数式接口,我们调用这个抽象函数式接口,但是我们不实现它,我们在调用的时候实现它是根据重量来比较,当然我们在调用的时候也可以根据颜色来比较
/**
举一反三:Predicate是传入一个参数进行判断,BiPredicate是传入两个参数进行判断,例如下面的例子
*/

private static List<Apple> filterAppleByBiPredicaTe(List<Apple> apples, BiPredicate<String, Long> predicate){
List<Apple> lists = new ArrayList()
for(Apple apple:apples){
	if (predicate.test(apple.getColor(),apple.getWeight())){
			lists.add(apple)
	}
}
//从上面的例子上来看我们知道BiPredicate传入两个参数,根据颜色和重量来进行比较,我们在调用的时候可以实现者两个参数的比较规则
}
登入後複製

Consumer 這個字中文意思是消費者,既然消費了就不會有回傳值

private static void printlnApple(List<Apple> list, Consumer<Apple> consumer){
        if (list.size()>0){
            for (Apple apple:list){
                consumer.accept(apple);
            }
        }
    }
//使用
printlnApple(listApple,apple -> System.out.println(apple.getColor()));
/**
从上面我们可以看到 这个Consumer不会返回任何对象
举一反三: 上面有BiPredicate,那么我们这个Consumer有没有BiConsumer,答案肯定是有的,下面我们来看下BiConsumer的用法
*/
private static void printGreenAppleByBiConsumer(List<Apple> apples, Consumer<Apple, String> consumer){
	if(apples.size()>0){
		for(Apple apple:apples){
			consumer.accept(apple, apple.getColor())
		}
	}
}
//使用
printGreenAppleByBiConsumer(list,(apple, color)->{
	if(apple.getWeight()>100&&color.equal("green")){
		System.out.println(apple);
		}
})
//学习玩上面的例子 我再提一个 LongConsumer,从字面上看,我们知道,它接收的是一个Long类型的,其他的并没什么区别,这里我们就不多啰嗦了
登入後複製

Function:函數,我們知道函數都是有輸出的,這裡的意思是說傳入一個參數,回傳一個參數,我們來看怎麼使用這個Function

 private static Long getWeightList(Apple apple, Function<Apple,Long> function)
    {
        return function.apply(apple);
    }
//使用
  Long green = getWeightList(new Apple("green", 120), Apple::getWeight);
        System.out.println(green);
//在这里我们可以看到,Function函数的apply抽象方法接收一个值,返回一个Long类型的值

//根据上面的经验,想必大家也已经知道了还有 BiFunction 这个函数式接口,没错它只是传进去的参数有两个而已,还是返回一个结果。除此之外还有呢,大家别着急,还有  IntFunction DoubleFunction,这些有什么作用呢,其实可以看成是Function的具体情况  IntFuntion也就是  apply里面的参数是Int类型的 DoubleFunction也就是apply的参数是double类型的,大家再看一下,是不是非常非常的简单呢
登入後複製

Supplier:這個字的意思是提供的意思,那也就是給你回傳數據,這個函數介面裡面會給你回傳數據,不需要你傳入資料

    private static Apple  getApple(Supplier<Apple> supplier){
       return  supplier.get();
    }
    //使用
        Apple green1 = getApple(() -> new Apple("green", 150));
        System.out.println(green1);
   //有人肯定再想了,这样创建对象那不是脱裤子放屁吗 多此一举,这里只是举一个小例子,在实际开发中肯定不会这样
登入後複製

3,Lamdba表達式之函數式推導

  • 1,透過一個類別的靜態方法去推斷

  • 2 ,用物件的實例方法去推斷

    private static void printstr(Consumer<String> consumer, String str){
        consumer.accept(str);
    }
    //使用
        Consumer<String> consumer = a -> System.out.println(a);
        printstr(consumer,"hello world");
   //写成
   		Consumer<String> consumer = System.out::println
   		printstr(consumer,"hello world");
   //思考我们为什么可以写成这样呢,我们看下println的源码
       // public final static PrintStream out = null;,在源码里面我们可以看到PrintStream是一个静态的类,而println是这个静态类下面的方法,因此可以做函数推导
       public void println(String x) {
       //刚好是接收一个参数,没有返回
        synchronized (this) {
            print(x);
            newLine();
        }
    }

```再举个例子

```java
//将字符串转换成数字
int  value = Integer.parseInt("123")
 Function<String, Integer> stringFunction = Integer::parseInt;
 Integer result = stringFunction.apply("123");
 System.out.println(result)
 /**
 很多人看到这里就很纳闷了,怎么冒出一个 Function来了呢,这就是类方法推导,parseInt是Integer的类方法它需要传入一个数,然后输出一个数,这样的模式就是前面我们介绍的Function函数式接口,接着再来看一个例子
 */
 String string = new String("hello");
 Function<Integer, Character> f = string::charAt;
 Character c = f3.apply(4);
 System.out.println(c);
登入後複製

建構函數推導:

Supplier<String> supplier = String::new
String s = supplier.get();
System.out.print(s)
//但是Apple是两个构造参数的,怎么办呢,还是有办法的
BiFunction<String, Long,Apple> appleFunction = Apple::new;
Apple apple = appleFunction.apply("red",100L);
//如果有更多的参数呢,在这里我们就可以自定义函数式接口
public interface CMyFuction<T,U,W,R>{
	R get(T t,U u, W w,R,r);
}
CMyFuction<String,String, Long> cmyfunction = Apple::new;
Apple apple =  cmyfunction.get("green","large",120);
//ok 这样自定义的函数式接口就完成了
登入後複製

範例:sort分類與函數式推導

//定义一个集合
  static List<Apple> listApple = Arrays.asList(
            new Apple("green",122),
            new Apple("red",  34),
            new Apple("black",135),
            new Apple("green",114),
            new Apple("yellow",54),
            new Apple("yellow",94));
 //以前我们排序的时候只知道调用 sort方法例如
 listAplle.sort()
 //但是我们不知道它怎么操作的,下面我们可以自定义排序的规则
 list.sort() //写道这里系统提示参数可以为Comparator,我们接着写
 list.sort(new Comparator<Apple>{
	public int compare(Apple o1, Apple o2){
		return o1.getColor().compareTo(o2.getColor());
	}
})
 //从上面的里例子来看我们是根据apple的颜色来进行比较的,当然我们也可以根据apple的weight来进行比较.接着我们进行函数推导
 list.sort(Comparator.comparing(Apple::getColor))
 //就这么一句完全代替了我们上面的功能,很明显简化了代码
登入後複製

以上是Java8 Lamdba函數式推導的語法怎麼表達的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:yisu.com
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板