Java開發筆記(六十二)如何定義函數式介面

来源:https://www.cnblogs.com/pinlantu/archive/2019/02/21/10415868.html
-Advertisement-
Play Games

前面介紹了Lambda表達式的用法,從實踐中發現它確實極大地方便了開發者,然而不管是匿名內部類還是Lambda表達式,所舉的例子都離不開各類數組的排序方法,倘使Lambda表達式僅能用於sort方法,無疑限制了它的應用範圍。那麼除了sort方法,還有哪些場景能夠將Lambda表達式派上用場呢?既然匿 ...


前面介紹了Lambda表達式的用法,從實踐中發現它確實極大地方便了開發者,然而不管是匿名內部類還是Lambda表達式,所舉的例子都離不開各類數組的排序方法,倘使Lambda表達式僅能用於sort方法,無疑限制了它的應用範圍。那麼除了sort方法,還有哪些場景能夠將Lambda表達式派上用場呢?既然匿名內部類與Lambda表達式都依附於某種介面,追根究底,就得好好研究一下這種介面的特別之處。
關於排序方法sort的第二個輸入參數,原本定義的參數類型是比較器Comparator,可是這個比較器真正有用的實乃唯一一個抽象方法compare。之前闡述Lambda表達式概念的時候,提到Lambda表達式指的是匿名方法,並且由於Java不支持把方法作為參數類型,因此只好再給方法加一層介面的包裝,於是sort方法里的參數類型變為Comparator介面而非compare方法了。
像Comparator這種掛羊頭賣狗肉的介面,錶面上是介面的結構,實際上給某個方法專用,為了有別於其它普通介面,它被Java稱作“函數式介面”。函數式介面擁有一般介面的形態,但其內部有且僅有一個抽象方法(方法也叫做函數),而這也是外部調用時採取Lambda表達式改寫的方法。除此之外,函數式介面還允許定義別的非抽象方法,包括預設方法與靜態方法。
搞清楚了函數式介面的來龍去脈,接下來不妨自定義一個全新的函數式介面。之前講到普通介面之時,定義了一個行為介面給各個動物類實現,這意味著行為動作的方法代碼與類定義代碼在一起定義。如果來了一個新的動物,就得提供對應的動物類定義及其動作代碼,日積月累各種動物類勢必越來越多。不過很多業務場景希望更靈活的邏輯,往往只要定義一個基礎的動物類,然後動物的每樣屬性都由成員方法讀寫,甚至動物的行為動作也由外部傳入。這樣可以制定一個“行動”方法,並通過“行為”介面包裝起來,再提供給動物類使用。下麵便是一個最簡單的行為介面代碼例子:

//定義一個行為介面,給動物類調用
public interface Behavior {

	// 聲明一個名叫行動的抽象方法
	public void act();
}

從上面的介面定義可知,Behavior介面有且僅有一個抽象方法act,因而它屬於函數式介面。接著編寫動物類的定義代碼,其中的midnight方法用來控制該動物在半夜乾什麼,具體的行動內容由輸入參數指定(參數類型為Behavior),具體的動物類代碼如下所示:

//演示動物類的定義,其中midnight方法的輸入參數為Behavior類型
public class Animal {

	// 定義一個名稱屬性
	private String name;

	public Animal(String name) {
		this.name = name;
	}
	
	public String getName() {
		return this.name;
	}
	
	// 定義一個半夜行動的方法。具體的動作由輸入行為的act方法執行
	public void midnight(Behavior behavior) {
		behavior.act();
	}
}

然後外部就能創建動物類Animal的實例,並調用該實例的midnight方法傳入規定的行為動作。以公雞為例,大公雞最喜歡在半夜雞叫了,那麼先創建一隻公雞實例,再命令它的midnight方法執行叫喚動作,這裡的叫喚動作若以匿名內部類書寫的話,可參考下列的調用代碼:

	// 測試公雞在半夜幹了啥
	private static void testCock() {
		Animal cock = new Animal("公雞");
		// 調用midnight方法時,傳入匿名內部類的實例
		cock.midnight(new Behavior() {
			@Override
			public void act() {
				System.out.println(cock.getName()+"在叫啦。");
			}
		});
	}

 

把以上的匿名內部類寫法改為Lambda表達式,將冗餘部分掐頭去尾簡化成瞭如下一行代碼:

		// 調用midnight方法時,傳入Lambda表達式的代碼。
		// 匿名方法不存在輸入參數的話,也要保留一對圓括弧占位子。
		cock.midnight(() -> System.out.println(cock.getName()+"在叫啦。"));

 

單單看這個Lambda表達式,姑且不論事實上的參數類型為何,至少在錶面上是把一段方法代碼作為輸入參數傳給了midnight。如此一來,函數式介面藉助Lambda表達式,成功地瞞天過海搖身變成了一種方法類型。
繼續演示其它動物,每當夜深人靜的時候,老貓便瞪圓眼睛出來捉老鼠了,於是往老貓實例的midnight方法輸入捉老鼠動作,相應的調用代碼如下所示:

	// 測試老貓在半夜幹了啥
	private static void testCat() {
		Animal cat = new Animal("老貓");
		// 調用midnight方法時,傳入Lambda表達式的代碼
		cat.midnight(() -> System.out.println(cat.getName()+"在捉老鼠。"));
	}

 

可見函數式介面結合Lambda表達式,將與行為有關的代碼減肥減得不能再瘦了。再奉上一段豬仔在半夜呼呼大睡的代碼例子:

	// 測試豬仔在半夜幹了啥
	private static void testPig() {
		Animal pig = new Animal("豬仔");
		// 調用midnight方法時,傳入Lambda表達式的代碼
		pig.midnight(() -> System.out.println(pig.getName()+"在呼呼大睡。"));
	}

 

最後運行上述三種動物的測試代碼,得到以下的日誌結果:

公雞在叫啦。
老貓在捉老鼠。
豬仔在呼呼大睡。

 

總結一下,函數式介面適用於外部把某個方法當作輸入參數的場合。通過利用函數式介面,一群相似的實體支持在調用之時單獨傳入個體動作,而無需像從前那樣派生出許多子類,還要在各個子類中分別實現它們的動作方法。



更多Java技術文章參見《Java開發筆記(序)章節目錄


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

-Advertisement-
Play Games
更多相關文章
  • 前言 開心一刻 一名劫匪慌忙中竄上了一輛車的後座,上車後發現主駕和副駕的一男一女疑惑地回頭看著他,他立即拔出槍威脅到:“趕快開車,甩掉後面的警車,否則老子一槍崩了你!”,於是副駕上的男人轉過臉對那女的說:“大姐,別慌,聽我口令把剛纔的動作再練習一遍,掛一檔,輕鬆離合,輕踩油門,走...走,哎 走.. ...
  • 題意 "題目鏈接" 有$n$個位置,每次你需要以$1 \sim n 1$的一個排列的順序去染每一個顏色,第$i$個數可以把$i$和$i+1$位置染成黑色。一個排列的價值為最早把所有位置都染成黑色的次數。問所有排列的分數之和 Sol 神仙題Orz 不難想到我們可以枚舉染色的次數$i \in [\lce ...
  • 《數據結構》這門課程的安排,就要開始各種演算法和數構的燒腦學習了,從最簡單的應用題型入手吧。 本題要求你寫個程式把給定的符號列印成沙漏的形狀。例如給定17個“*”,要求按下列格式列印 所謂“沙漏形狀”,是指每行輸出奇數個符號;各行符號中心對齊;相鄰兩行符號數差2;符號數先從大到小順序遞減到1,再從小到 ...
  • 在做關於NIO TCP編程小案例時遇到無法監聽write的問題,沒想到只是我的if語句的位置放錯了位置,哎,看了半天沒看出來 貼下課堂筆記: 在Java中使用NIO進行網路TCP套接字編程主要以下幾個類: ServerSocketChannel: 服務端套接字通道,主要監聽接收客戶端請求 Selec ...
  • wxPython框架雖然成熟穩定,但是相對最近更火的PyQt框架來說,還是顯得古老了一些,控制項風格不符合現代審美觀,因此痞子衡決定學習一下PyQt的用法,感受下PyQt做出來的界面效果到底如何。根據wxPython學習經驗,當然首先要從PyQt的可視化GUI構建工具Qt Designer開始下手,因... ...
  • 本文通過講解如何解析application.properties屬性,介紹了幾個註解的運用@Value @ConfigurationProperties @EnableConfigurationProperties @Autowired @ConditionalOnProperty ...
  • 一個項目里只能有一個main函數, 如果出現 error:LNK2005 的錯誤,那麼需要檢查你是不是有兩個源代碼文件中都定義了main函數。 例如: 如果在a.cpp中定義了main函數,在b.cpp中也定義main函數,編譯執行就會報 error:LNK2005 的錯誤。 ...
  • 一、 lucene簡介 1. Lucene Lucene是apache下的一個開源的全文檢索引擎工具包。它為軟體開發人員提供一個簡單易用的工具包(類庫),以方便的在目標系統中實現全文檢索的功能。 官網: http://lucene.apache.org/ 2. 全文檢索 全文檢索是指電腦索引程式通 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...