Java-->Lambda表達式

来源:http://www.cnblogs.com/xmcx1995/archive/2016/08/18/5785000.html
-Advertisement-
Play Games

一、函數式介面 函數式介面(functional interface 也叫功能性介面,其實是同一個東西)。簡單來說,函數式介面是只包含一個方法的介面。比如Java標準庫中的java.lang.Runnable和 java.util.Comparator都是典型的函數式介面。java 8提供 @Fun ...


一、函數式介面

函數式介面(functional interface 也叫功能性介面,其實是同一個東西)。簡單來說,函數式介面是只包含一個方法的介面。比如Java標準庫中的java.lang.Runnable和 java.util.Comparator都是典型的函數式介面。
java 8提供 @FunctionalInterface作為註解,這個註解是非必須的,只要介面符合函數式介面的標準(即只包含一個方法的介面),虛擬機會自動判斷, 但 最好在介面上使用註解@FunctionalInterface進行聲明,以免團隊的其他人員錯誤地往介面中添加新的方法。

Java中的lambda無法單獨出現,它需要一個函數式介面來盛放,lambda表達式方法體其實就是函數介面的實現.

下麵的介面就是一個函數式介面

package com.yztcedu.lambdademo_01;

@FunctionalInterface //添加此註解後,介面中只能有一個抽象方法。
public interface A {
	void call();

}

二、lambda語法

包含三部分:
1、一個括弧內用逗號分隔的形式參數,參數是函數式介面裡面方法的參數
2、一個箭頭符號:->
3、方法體,可以是表達式和代碼塊。

(parameters) -> expression 或者 (parameters) -> { statements; }

通過下麵的代碼可以看到lambda表達式設計的代碼更簡潔,而且可讀性更好。

package com.yztcedu.lambdademo_01;

public class Demo1 {
	public static void main(String[] args) {
		runThreadByLambda();
		runThreadByInnerClass();
	}

	public static void runThreadByLambda() {
		/*
		 Runnable就是一個函數式介面:他只有一個方法run()方法。
		 1、因為run()方法沒有參數,所以   ->前面的()中不需要聲明形參
		 2、run返回的是void,所以不需要return。
		 3、->後面寫的代碼其實就是定義在run方法內的代碼。因為此處代碼只有一行,所以{}也可以省略。如果此處多與一行,則無法省略。
		 */
		Runnable runnable = () -> System.out.println("這個是用拉姆達實現的線程");
		new Thread(runnable).start();
	}

	public static void runThreadByInnerClass() {
		Runnable runnable = new Runnable() {

			@Override
			public void run() {
				System.out.println("這個是用內部類實現的線程");

			}
		};
		new Thread(runnable).start();
	}
}

三、方法引用

其實是lambda表達式的一種簡化寫法。所引用的方法其實是lambda表達式的方法體實現,語法也很簡單,左邊是容器(可以是類名,實例名),中間是"::",右邊是相應的方法名。如下所示:

ObjectReference::methodName

一般方法的引用格式:

  1. 如果是靜態方法,則是ClassName::methodName。如 Object ::equals
  2. 如果是實例方法,則是Instance::methodName。如Object obj=new Object();obj::equals;
  3. 構造函數.則是ClassName::new
package com.yztcedu.lambdademo_01;

public class Demo2 {
	public static void main(String[] args) {
		/*
		 * 方法引用
		 */
		Runnable runnable = Demo2::run;
		new Thread(runnable).start();
	}
	
	
	public static void run(){
		System.out.println("方法引用的代碼...");
	}

}

可以看出,doSomething方法就是lambda表達式的實現,這樣的好處就是,如果你覺得lambda的方法體會很長,影響代碼可讀性,方法引用就是個解決辦法

四、預設方法---介面改進

簡單說,就是介面可以有實現方法,而且不需要實現類去實現其方法。只需在方法名前面加個default關鍵字即可。

package com.yztcedu.lambdademo_01;

@FunctionalInterface
public interface A {
	void call();

	default void fun() {
		System.out.println("我是介面的預設方法1中的代碼");
	}

	default void fun2() {
		System.out.println("我是介面的預設方法2中的代碼");
	}

}

為什麼要有這個特性?首先,之前的介面是個雙刃劍,好處是面向抽象而不是面向具體編程,缺陷是,當需要修改介面時候,需要修改全部實現該介面的類,目前的 java 8之前的集合框架沒有foreach方法,通常能想到的解決辦法是在JDK里給相關的介面添加新的方法及實現。然而,對於已經發佈的版本,是沒法在給介面 添加新方法的同時不影響已有的實現。所以引進的預設方法。他們的目的是為了使介面沒有引入與現有的實現不相容發展。

java8中介面和抽象類的區別

形同點:
1.都是抽象類型;
2.都可以有實現方法(以前介面不行);
3.都可以不需要實現類或者繼承者去實現所有方法,(以前不行,現在介面中預設方法不需要實現者實現)

不同點
1.抽象類不可以多重繼承,介面可以(無論是多重類型繼承還是多重行為繼承);
2.抽象類和介面所反映出的設計理念不同。其實抽象類表示的是"is-a"關係,介面表示的是"like-a"關係;
3.介面中定義的變數預設是public static final 型,且必須給其初值,所以實現類中不能重新定義,也不能改變其值;抽象類中的變數預設是 default 型,其值可以在子類中重新定義,也可以重新賦值。

總結:預設方法給予我們修改介面而不破壞原來的實現類的結構提供了便利,目前java 8的集合框架已經大量使用了預設方法來改進了,當我們最終開始使用Java 8的lambdas表達式時,提供給我們一個平滑的過渡體驗。也許將來我們會在API設計中看到更多的預設方法的應用。

五、使用lambda改進的集合框架

5.1 集合中內部迭代

package com.yztcedu.lambdademo_01;

import java.util.ArrayList;
import java.util.List;

public class Demo3 {
	public static void main(String[] args) {
		List<User> users = new ArrayList<User>();
		users.add(new User(20, "張三"));
		users.add(new User(22, "李四"));
		users.add(new User(10, "王五"));

		users.forEach((User user) -> System.out.println(user.getAge()));

	}

}

5.2 Stream API

流(Stream)僅僅代表著數據流,並沒有數據結構,所以他遍歷完一次之後便再也無法遍歷(這點在編程時候需要註意,不像Collection,遍歷多少次裡面都還有數據),它的來源可以是Collection、array、io等等。

流作用是提供了一種操作大數據介面,讓數據操作更容易和更快。它具有過濾、映射以及減少遍曆數等方法,這些方法分兩種:中間方法和終端方法,“流”抽象天生就該是持續的,中間方法永遠返回的是Stream,因此如果我們要獲取最終結果的話,必須使用終點操作才能收集流產生的最終結果。區分這兩個方法是看他的返回值,如果是Stream則是中間方法,否則是終點方法。

filter

在數據流中實現過濾功能是首先我們可以想到的最自然的操作了。Stream介面暴露了一個filter方法,它可以接受表示操作的Predicate實現來使用定義了過濾條件的lambda表達式。

import java.util.stream.Stream;

public class StreamDemo {
	public static void main(String[] args) {
		List<User> users = new ArrayList<User>();
		users.add(new User(20, "張三"));
		users.add(new User(22, "李四"));
		users.add(new User(10, "王五"));
		
		Stream<User> stream = users.stream();
		stream.filter(p -> p.getAge() > 20); //過濾年齡大於20的
	}

}

map

假使我們現在過濾了一些數據,比如轉換對象的時候。Map操作允許我們執行一個Function的實現(Function<T,R>的泛型T,R分別表示執行輸入和執行結果),它接受入參並返回。

package com.yztcedu.lambdademo_01;

import java.util.ArrayList;
import java.util.List;
import java.util.stream.Stream;

public class StreamDemo {
	public static void main(String[] args) {
		List<User> users = new ArrayList<User>();
		users.add(new User(20, "張三"));
		users.add(new User(22, "李四"));
		users.add(new User(10, "王五"));
		
		Stream<User> stream = users.stream();
		 //所有的年齡大於20歲的User對象,轉換為字元串50對象。現在流中只有字元串對象了。
		stream.filter((User user) ->  user.getAge() > 20).map((User user) -> {return "50";});
		
	}

}

count

count方法是一個流的終點方法,可使流的結果最終統計,返回long

package com.yztcedu.lambdademo_01;

import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collector;
import java.util.stream.Stream;

public class StreamDemo {
	public static void main(String[] args) {
		List<User> users = new ArrayList<User>();
		users.add(new User(20, "張三"));
		users.add(new User(22, "李四"));
		users.add(new User(10, "王五"));
		
		
		Stream<User> stream = users.stream();
		long count = stream.filter((User user) ->  user.getAge() >= 20).map((User user) -> {return "50";})
		.count(); //返迴流中元素的個數。
		System.out.println(count);
		
		
		
	}

}

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

-Advertisement-
Play Games
更多相關文章
  • 電腦的發展歷史 電腦的學名叫電腦,電腦是用來做計算的。在古時候,人們最早使用的計算工具可能是手指,英文單詞“digit”既有“數字”的意思,又有“手指“的意思。古人用石頭打獵,所以還有可能是石頭來輔助計算。 缺點:手指和石頭太低效了 後來出現了”結繩 “記事。 缺點:結繩慢,繩子還有長度限制。 又 ...
  • 這一篇首先從allitebooks.com里抓取書籍列表的書籍信息和每本書對應的ISBN碼。 一、分析需求和網站結構 allitebooks.com這個網站的結構很簡單,分頁+書籍列表+書籍詳情頁。 要想得到書籍的詳細信息和ISBN碼,我們需要遍歷所有的頁碼,進入到書籍列表,然後從書籍列表進入到每本 ...
  • 1. 基本的代碼結構為: 2. ...
  • 1、 set(集合)——包含了經過排序了的數據,這些數據的值(value)必須是唯一的。 也就是說輸入set容器後得到數據,會去重併排序。 s.insert()插入一個元素 s.begin() s.end()分別返迴首尾指針 s.clear() 清空集合 遍歷需要利用迭代器set<類型>::iter ...
  • 程式主文件標誌if __name__=="__main__": 在程式執行python 1.py 時候 程式1.py __name__ 為 main調用其他文件是,__name__ 為程式自己名字等於__name__ 一切事物都是對象,對象是有類創建的 int 內部功能a = 95b = 10c = ...
  • --> 這裡用到兩種方法...其實也不算兩種,就一點點不一樣而已... > Test 測試類 --> MyThread類即線程實現類 --> 邪惡的分割線 --> Test測試 --> MyThread線程實現類 --> 感覺第二種也完全是多餘的啊,就是一種方法... ...
  • 最近看了看socket網路編程,對於我這種一點經驗都沒有的選手來說只能理解一點點吧。所以在此記錄一下最近的收穫。 socket編程無非就那幾個函數,對於服務端來說,主要的為socket(),bind(),listen(),accept(),close()。對於客戶端來說主要為connect(),cl ...
  • 1、java.util.Collection 是一個集合介面。它提供了對集合對象進行基本操作的通用介面方法。Collection介面在Java 類庫中有很多具體的實現。Collection介面的意義是為各種具體的集合提供了最大化的統一操作方式。 Collection ├List │├LinkedLi ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...