Lambda表達式---Day27

来源:https://www.cnblogs.com/hpcz190911/archive/2019/12/03/11974325.html
-Advertisement-
Play Games

函數式編程思想概述 在數學中,函數就是有輸入量、輸出量的一套計算方案,也就是“拿什麼東西做什麼事情”。相對而言,面向對象過 分強調“必須通過對象的形式來做事情”,而函數式思想則儘量忽略面向對象的複雜語法——強調做什麼,而不是以 什麼形式做。 面向對象的思想: 做一件事情,找一個能解決這個事情的對象, ...


函數式編程思想概述

  在數學中,函數就是有輸入量、輸出量的一套計算方案,也就是“拿什麼東西做什麼事情”。相對而言,面向對象過 分強調“必須通過對象的形式來做事情”,而函數式思想則儘量忽略面向對象的複雜語法——強調做什麼,而不是以 什麼形式做。

  面向對象的思想: 做一件事情,找一個能解決這個事情的對象,調用對象的方法,完成事情.

  函數式編程思想: 只要能獲取到結果,誰去做的,怎麼做的都不重要,重視的是結果,不重視過程

冗餘的Runnable代碼

  傳統寫法:當需要啟動一個線程去完成任務時,通常會通過 java.lang.Runnable 介面來定義任務內容,並使用 java.lang.Thread 類來啟動該線程

 1 package demosummary.lambda;
 2 
 3 public class RunnableDemo1 {
 4     public static void main(String[] args) {
 5         //匿名內部類
 6         Runnable task = new Runnable() {
 7             @Override
 8             public void run() {
 9                 System.out.println("多線程任務執行");
10             }
11         };
12         new Thread(task).start();//啟動線程
13     }
14 }

  對上述代碼分析:

    1.Thread 類需要 Runnable 介面作為參數,其中的抽象 run 方法是用來指定線程任務內容的核心

    2.為了指定 run 的方法體,不得不需要 Runnable 介面的實現類

    3.為了省去定義一個 RunnableImpl 實現類的麻煩,不得不使用匿名內部類; 

    4.必須覆蓋重寫抽象 run 方法,所以方法名稱、方法參數、方法返回值不得不再寫一遍,且不能寫錯; 

    5.而實際上,似乎只有方法體才是關鍵所在

lambda更優的寫法

 1 package demosummary.lambda;
 2 
 3 /**
 4  * Lambda寫法
 5  */
 6 public class LambdaDemo1 {
 7     public static void main(String[] args) {
 8         new Thread(() -> System.out.println("多線程任務執行")).start();//啟動線程
 9     }
10 }

  以上代碼和上一個代碼是同樣的執行效果,lambda寫法需要在1.8以上的編譯才能通過

回顧匿名內部類

  1.使用實現類

    要啟動一個線程,需要創建一個 Thread 類的對象並調用 start 方法。而為了指定線程執行的內容,需要調用 Thread 類的構造方法:public Thread(Runnable target)

      為了獲取 Runnable 介面的實現對象,可以為該介面定義一個實現類 RunnableImpl :

1 package demosummary.lambda;
2 
3 public class RunnableImpl implements Runnable{
4     @Override
5     public void run() {
6         System.out.println("多線程任務執行");
7     }
8 }

    然後創建該實現類的對象作為 Thread 類的構造參數:

 1 package demosummary.lambda;
 2 
 3 public class RunnableImplTest {
 4     public static void main(String[] args) {
 5         //創建RunnableImpl對象
 6         RunnableImpl ri = new RunnableImpl();
 7         //啟動線程
 8         new Thread(ri).start();
 9     }
10 }

  2.使用匿名內部類

    這個 RunnableImpl 類只是為了實現 Runnable 介面而存在的,而且僅被使用了唯一一次,所以使用匿名內部類的 語法即可省去該類的單獨定義,即匿名內部類:

 1 package demosummary.lambda;
 2 
 3 /**
 4  * 使用匿名內部類
 5  */
 6 public class RunnableImplInit {
 7     public static void main(String[] args) {
 8         new Thread(new Runnable() {
 9             @Override
10             public void run() {
11                 System.out.println("多線程任務執行");
12             }
13         }).start();
14     }
15 }

  對上述匿名內部類方法分析

    仔細分析該代碼中的語義, Runnable 介面只有一個 run 方法的定義:public abstract void run();

    無參數:不需要任何條件即可執行該方案。

    無返回值:該方案不產生任何結果。

    代碼塊(方法體):該方案的具體執行步驟

  而使用lambda語法要更加簡單:

    () ‐> System.out.println("多線程任務執行!")

      前面的一對小括弧即 run 方法的參數(無),代表不需要任何條件;

      中間的一個箭頭代表將前面的參數傳遞給後面的代碼;

      後面的輸出語句即業務邏輯代碼。

  匿名內部類的好處與弊端 

    一方面,匿名內部類可以幫我們省去實現類的定義;另一方面,匿名內部類的語法——確實太複雜了

Lambda表達式的標準格式

  Lambda省去面向對象的條條框框,格式由3個部分組成:

    1.一些參數 2.一個箭頭 3.一段代碼

  Lambda表達式的標準格式為:(參數類型 參數名稱) ‐> { 代碼語句 }

  格式說明:

    1.小括弧內的語法與傳統方法參數列表一致:無參數則留空;

    2.多個參數則用逗號分隔。 -> 是新引入的語法格式,代表指向動作。

    3.大括弧內的語法與傳統方法體要求基本一致。 

Lambda省略格式 

  省略規則:

    1. 小括弧內參數的類型可以省略;

    2. 如果小括弧內有且僅有一個參,則小括弧可以省略;

    3. 如果大括弧內有且僅有一個語句,則無論是否有返回值,都可以省略大括弧、return關鍵字及語句分號

Lambda的使用前提

  Lambda的語法非常簡潔,完全沒有面向對象複雜的束縛。但是使用時有幾個問題需要特別註意

    1. 使用Lambda必須具有介面,且要求介面中有且僅有一個抽象方法。 無論是JDK內置的 Runnable 、 Comparator 介面還是自定義的介面,只有當介面中的抽象方法存在且唯一 時,才可以使用Lambda。

    2. 使用Lambda必須具有上下文推斷。 也就是方法的參數或局部變數類型必須為Lambda對應的介面類型,才能使用Lambda作為該介面的實例。

    備註:有且僅有一個抽象方法的介面,稱為“函數式介面”。

使用lambda標準格式(無參無返回)

  給定一個廚子 Cook 介面,內含唯一的抽象方法 makeFood ,且無參數、無返回值。如下:

1 package demosummary.lambda;
2 
3 /**
4  * 給定一個廚子 Cook 介面,內含唯一的抽象方法 makeFood ,且無參數、無返回值。如下:
5  */
6 public interface Cook {
7     void makeFood();
8 }

  在下麵的代碼中,請使用Lambda的標準格式調用 invokeCook 方法,列印輸出“吃飯啦!”字樣:

 1 package demosummary.lambda;
 2 
 3 /**
 4  * 在下麵的代碼中,請使用Lambda的標準格式調用 invokeCook 方法,列印輸出“吃飯啦!”字樣:
 5  */
 6 public class InvokeCook {
 7     public static void main(String[] args) {
 8 //        invokeCook(() -> {
 9 //            System.out.println("吃飯了");
10 //        });
11         
12         //省略格式
13         invokeCook(() -> System.out.println("吃飯了"));
14     }
15 
16     private static void invokeCook(Cook cook) {
17         cook.makeFood();
18     }
19 }

  備註:小括弧代表 Cook 介面 makeFood 抽象方法的參數為空,大括弧代表 makeFood 的方法體。 

使用Lambda標準格式(有參有返回)

  給定一個計算器 Calculator 介面,內含抽象方法 calc 可以將兩個int數字相加得到和值:

1 package demosummary.lambda;
2 
3 public interface Calculator {
4     int cal(int a, int b);
5 }

  在下麵的代碼中,請使用Lambda的標準格式調用 invokeCalc 方法,完成120和130的相加計算:

 1 package demosummary.lambda;
 2 
 3 public class invokeCal {
 4     public static void main(String[] args) {
 5 //        invokeCal(120,130,(int a , int b) -> {
 6 //            return a + b;
 7 //        });
 8 
 9         //省略格式
10         invokeCal(120,130,(int a , int b) -> a + b);
11     }
12 
13     public static void invokeCal(int a , int b , Calculator calculator){
14         int result = calculator.cal(a, b);
15         System.out.println("輸出結果:"+result);
16     }
17 }

Lambda的參數和返回值

  需求:     1.使用數組存儲多個Person對象     2.對數組中的Person對象使用Arrays的sort方法通過年齡進行升序排序

 1 package demosummary.lambda;
 2 
 3 public class Person {
 4     private String name;
 5     private int age;
 6 
 7     public Person() {
 8     }
 9 
10     public Person(String name, int age) {
11         this.name = name;
12         this.age = age;
13     }
14 
15     public String getName() {
16         return name;
17     }
18 
19     public void setName(String name) {
20         this.name = name;
21     }
22 
23     public int getAge() {
24         return age;
25     }
26 
27     public void setAge(int age) {
28         this.age = age;
29     }
30 
31     @Override
32     public String toString() {
33         return "Person{" +
34                 "name='" + name + '\'' +
35                 ", age=" + age +
36                 '}';
37     }
38 }

 

 1 package demosummary.lambda;
 2 
 3 import java.util.Arrays;
 4 
 5 public class LambdaComparator {
 6     public static void main(String[] args) {
 7         Person[] array = {
 8                 new Person("古力娜扎",22),
 9                 new Person("迪麗熱巴",21),
10                 new Person("歐陽娜娜",19)
11         };
12 
13 //        Arrays.sort(array,(Person a, Person b)->{
14 //            return a.getAge() - b.getAge();
15 //        });
16 
17         //省略格式
18         Arrays.sort(array,(Person a, Person b) -> a.getAge() - b.getAge());
19 
20         for (Person person : array) {
21             System.out.println(person);
22         }
23     }
24 }

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • javascript 報錯 string.split is not a function ...
  • 自定義html元素滑鼠右鍵菜單 實現思路 在觸發contextmenu事件時,取消預設行為(也就是阻止瀏覽器顯示自帶的菜單),獲取右鍵事件對象,來確定滑鼠的點擊位置,作為顯示菜單的left和top值 編碼實現 <!DOCTYPE html> <html> <head> <meta charset=" ...
  • 功能:在文本框中輸入待辦事項按下回車後,事項會出現在未完成列表中;點擊未完成事項前邊的覆選框後,該事項會出現在已完成列表中,反之亦然;點擊刪除按鈕會刪除該事項。待辦事項的數據是保存到本地存儲的(localStorage),就算關閉頁面再打開,數據還是存在的(前提是要用相同瀏覽器)。 ToDoList ...
  • 基於Taro多端實踐TaroPop:自定義模態框|dialog對話框|msg消息框|Toast提示 taro自定義彈出框支持編譯到多端H5/小程式/ReactNative,還可以自定義彈窗類型/彈窗樣式、多按鈕事件/樣式、自動關閉、遮罩層、彈窗顯示位置及自定義內容模板 用法 ▍在相應頁面引入組件 i ...
  • 創建對象三種方式: 調用系統的構造函數創建對象 自定義構造函數創建對象(結合第一種和需求通過工廠模式創建對象) 字面量的方式創建對象 第一種:調用系統的構造函數創建對象 //小蘇舉例子: //實例化對象 var obj = new Object(); //對象有特征 屬性;和 行為 方法 //添加屬 ...
  • 功能描述: 當滾動條滑到某個位置時,顯示電梯導航; 當用戶滾動滾動條時,讓電梯導航的選中狀態和當前滾動到的區域保持一致; 當用戶點擊電梯導航時,滾動條滾動到被點擊導航對應的區域 準備工作: 首先將jQuery文件以及你自己的js文件引入你的html里,jq文件要放在上面 <script src="j ...
  • 6.分散式事務解決方案之最大努力通知 6.1. 什麼是最大努力通知 最大努力通知也是一種解決分散式事務的方案,下邊是一個是充值的例子:交互流程 :1、賬戶系統調用充值系統介面2、充值系統完成支付處理向賬戶系統發起充值結果通知若通知失敗,則充值系統按策略進行重覆通知3、賬戶系統接收到充值結果通知修改充 ...
  • 工作這麼久了,對於Java中時間日期的操作一直很蛋疼,一會用Date,一會用Calendar一會用LocalDateTime,始終沒有認真總結過它們的聯繫與區別。迷迷糊糊用了好幾年了,今天終於搞清楚了! ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...