高階函數(shù)和Java的Lambda

2017年的第一天,我坐在獨(dú)墅湖邊,寫下這篇文章。


獨(dú)墅湖.jpeg

在數(shù)學(xué)和計(jì)算機(jī)科學(xué)中,高階函數(shù)是至少滿足下列一個(gè)條件的函數(shù):

  • 接受一個(gè)或多個(gè)函數(shù)作為輸入
  • 輸出一個(gè)函數(shù)

java世界迎來(lái)新的一等公民——函數(shù)

java 8引入了函數(shù)式編程。函數(shù)式編程重點(diǎn)在函數(shù),函數(shù)變成了Java世界里的一等公民,函數(shù)和其他值一樣,可以到處被定義,可以作為參數(shù)傳入另一個(gè)函數(shù),也可以作為函數(shù)的返回值,返回給調(diào)用者。利用這些特性,可以靈活組合已有函數(shù)形成新的函數(shù),可以在更高層次上對(duì)問(wèn)題進(jìn)行抽象。

使用高階函數(shù)之前的求和、求平方和、求立方和的寫法:

public class TestHighOrderFunction {
 
   public static int identity(int x) {
      return x;
   }

   public static int sum_integers(int a, int b) {
     int sum = 0;
     for (int i = a; i <= b; i++) {
       sum += identity(i);
     }
    return sum;
   }

   public static int square(int x) {
      return x * x;
   }

  public static int sum_square(int a, int b) {
     int sum = 0;
     for (int i = a; i <= b; i++) {
        sum += square(i);
     }
     return sum;
   }

   public static double cube(int x) {
      return x * x * x;
   }

   public static int sum_cubes(int a, int b) {
      int sum = 0;
      for (int i = a; i <= b; i++) {
         sum += cube(i);
      }
      return sum;
   }

    public static void main(String[] a) {

        System.out.println(sum_integers(1, 10)); // return 55
        System.out.println(sum_square(1, 10));  // return 385
        System.out.println(sum_cubes(1, 10));  // return 3025
    }
}

我們發(fā)現(xiàn)sum_開(kāi)頭的方法里,代碼很類似,三者唯一區(qū)別在于

   sum += identity(i);
   sum += square(i);
   sum += cube(i);

在軟件工程里有DRY(don't repeat yourself )的準(zhǔn)則。我們來(lái)看看使用高階函數(shù)怎樣優(yōu)化剛才的這些代碼:

 public interface Function {
     int opera(int a);
 }

 public static void main(String[] a) {
  
    Function identity = x->x;
    Function square = x->x*x;
    Function cube = x -> x*x*x;
    System.out.println(sum(identity, 1,10)); // return 55
    System.out.println(sum(square, 1,10)); // return 385
    System.out.println(sum(cube, 1,10));   // return 3025
 }
 
 public static int sum(Function term, int a, int b) {
  
     int sum = 0;
     for (int i = a; i <= b; i++) {
        sum += term.opera(i);
     }
     return sum;
 }

得到的結(jié)果,跟上面的TestHighOrderFunction類中運(yùn)行的結(jié)果是一樣的。不過(guò),這里的sum方法中使用了

   sum += term.opera(i);

取代了原先的代碼。term.opera(i)對(duì)應(yīng)的是原先identity(i)、square(i)、cube(i),在這里Function函數(shù)被當(dāng)做參數(shù)進(jìn)行傳遞。這就是高階函數(shù)的特性。

對(duì)于for循環(huán),我們還能用更優(yōu)雅的方式進(jìn)行優(yōu)化,下面使用了遞歸的方式。

 public interface Function {
     int opera(int a);
 }

 public static void main(String[] a) {
  
      Function identity = x->x;
      Function square = x->x*x;
      Function cube = x -> x*x*x;
      Function inc = x->x+1; // 定義next函數(shù)
      System.out.println(sum(identity, 1,inc,10)); // return 55
      System.out.println(sum(square, 1,inc,10)); // return 385
      System.out.println(sum(cube, 1,inc,10));   // return 3025
 }
 
 public static int sum(Function term, int a, Function next, int b) {
  
      if (a>b) {
          return 0;
      } else {
          return term.opera(a) + sum(term, next.opera(a), next, b);
      }
 }

@FunctionalInterface

在java 8之前我們使用Thread,可能是這樣的

   new Thread(new Runnable() {    
        public void run() {        
              System.out.println("test");    
       }
   }).start();

由于Java 8引入了lambda表達(dá)式,我們可以這樣寫

   new Thread(()->System.out.println("test")).start();

lambda表達(dá)式源于lambda演算。

Lambda演算可以被稱為最小的通用程序設(shè)計(jì)語(yǔ)言。它包括一條變換規(guī)則(變量替換)和一條函數(shù)定義方式,Lambda演算之通用在于,任何一個(gè)可計(jì)算函數(shù)都能用這種形式來(lái)表達(dá)和求值。因而,它是等價(jià)于圖靈機(jī)的。盡管如此,Lambda演算強(qiáng)調(diào)的是變換規(guī)則的運(yùn)用,而非實(shí)現(xiàn)它們的具體機(jī)器。可以認(rèn)為這是一種更接近軟件而非硬件的方式。

我們點(diǎn)擊Runnable的源碼時(shí),發(fā)現(xiàn)Runnable使用了@FunctionalInterface,這在java 8之前是沒(méi)有的。

@FunctionalInterface
public interface Runnable {
    /**
     * When an object implementing interface <code>Runnable</code> is used
     * to create a thread, starting the thread causes the object's
     * <code>run</code> method to be called in that separately executing
     * thread.
     * <p>
     * The general contract of the method <code>run</code> is that it may
     * take any action whatsoever.
     *
     * @see     java.lang.Thread#run()
     */
    public abstract void run();
}

@FunctionalInterface是Java 8為函數(shù)式接口引入的一個(gè)新的注解。表明該接口是函數(shù)式接口,它只包含唯一一個(gè)抽象方法。任何可以接受一個(gè)函數(shù)式接口實(shí)例的地方,都可以用lambda表達(dá)式。

我們?cè)賮?lái)看一個(gè)匿名函數(shù)的例子。

button.setOnClickListener(new Button.OnClickListener(){   
     public void onClick(View v) {        
         Log.i(TAG,"點(diǎn)擊button");    
     }
});

我們將匿名函數(shù)改成lambda表達(dá)式

button.setOnClickListener((v)-> Log.i(TAG,,"點(diǎn)擊button"));

這樣改造的好處在于,lambda對(duì)象的創(chuàng)建是通過(guò)字節(jié)碼指令invokedynamic來(lái)完成的,減少了類型和實(shí)例的創(chuàng)建消耗。而匿名類需要新的對(duì)象的創(chuàng)建。

JDK中的函數(shù)式接口舉例

java.lang.Runnable,
java.awt.event.ActionListener,
java.util.Comparator,
java.util.concurrent.Callable
java.util.function包下的接口,如Consumer、Predicate、Supplier等

簡(jiǎn)化的lambda——方法引用(Method Reference)

lambda已經(jīng)簡(jiǎn)化了代碼的寫法,然而方法引用進(jìn)一步簡(jiǎn)化了lambda的寫法。
方法引用的使用方式:類名::方法名

類型 使用方式 備注
引用靜態(tài)方法 ContainingClass::staticMethodName Integer::valueOf簡(jiǎn)化了i->Integer.valueOf(i)的寫法
引用特定對(duì)象的實(shí)例方法 containingObject::instanceMethodName s::toString()簡(jiǎn)化了()->s.toString()
引用特定類型的任意對(duì)象的實(shí)例方法 ContainingType::methodName System.out::println簡(jiǎn)化了(s)->System.out.println(s),其中System.out表示的是PrintStream對(duì)象
引用構(gòu)造函數(shù) ClassName::new String::new簡(jiǎn)化了()->new String()

我們來(lái)看一個(gè)簡(jiǎn)單的例子,對(duì)User按照name來(lái)進(jìn)行排序,最初我們會(huì)這樣寫。

  User u1 = new User("tony");
  User u2 = new User("cafei");
  User u3 = new User("aaron");
  
  List<User> users = Arrays.asList(u1,u2,u3);
  
  Collections.sort(users, new Comparator<User>(){

   @Override
   public int compare(User u1, User u2) {
    return u1.getName().compareTo(u2.getName());
   }
   
  });

在java 8以后,Comparator增加了一個(gè)靜態(tài)方法comparing(Function<? super T, ? extends U> keyExtractor),我們可以把排序的寫法簡(jiǎn)化成這樣:

Collections.sort(users, Comparator.comparing((User u)->u.getName()));

如果使用方法引用,還可以更加簡(jiǎn)化代碼

Collections.sort(users,Comparator.comparing(User::getName));

集合中的應(yīng)用

在java 8中可以使用新增的api Streams來(lái)操作集合,Streams是區(qū)別于java.io包里的InputStream 和 OutputStream的概念,是對(duì)集合功能的增強(qiáng)。如果你曾經(jīng)了解過(guò)Scala、RxJava等函數(shù)式編程,那么看了它的語(yǔ)法以后一定會(huì)覺(jué)得似曾相識(shí)。我們來(lái)看兩段代碼,看看它是如何使用的。

List<Integer> list = Arrays.asList(1, 2, 3, 5, 7, 9, 10)
    .stream()
    .filter(i -> i >= 5)
    .collect(Collectors.toList());

System.out.println("list=" + list); // return list=[5, 7, 9, 10]
  Arrays.asList("tony", "cafei", "aaron")
           .stream()
           .map(str -> str.toUpperCase())
           .forEach(it -> System.out.println(it));

上面的代碼還可以使用方法引用的方式:

  Arrays.asList("tony", "cafei", "aaron")
        .stream()
        .map(String::toUpperCase)
        .forEach(System.out::println);

使用這樣的鏈?zhǔn)秸{(diào)用非常cool。而且,map、filter等方法都是高階函數(shù)。

寫在最后

lambda是java 8最為重要的特性,lambda表達(dá)式并非只是Java的語(yǔ)法糖,而是由編譯器和JVM共同配合來(lái)實(shí)現(xiàn)的。自從使用了lambda以后我感覺(jué)再也回不去了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,119評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,382評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 176,038評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 62,853評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,616評(píng)論 6 408
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 55,112評(píng)論 1 323
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,192評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 42,355評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,869評(píng)論 1 334
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,727評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,928評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,467評(píng)論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,165評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 34,570評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 35,813評(píng)論 1 282
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,585評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,892評(píng)論 2 372

推薦閱讀更多精彩內(nèi)容