Java函數式編程原理以及應用

来源:https://www.cnblogs.com/wenbochang/archive/2019/08/22/11385491.html
-Advertisement-
Play Games

一. 函數式編程 Java8所有的新特性基本基於函數式編程的思想,函數式編程的帶來,給Java註入了新鮮的活力。 下麵來近距離觀察一下函數式編程的幾個特點: 函數可以作為變數、參數、返回值和數據類型。 基於表達式來替代方法的調用 函數無狀態,可以併發和獨立使用 函數無副作用,不會修改外部的變數 函數 ...


一. 函數式編程

Java8所有的新特性基本基於函數式編程的思想,函數式編程的帶來,給Java註入了新鮮的活力。

下麵來近距離觀察一下函數式編程的幾個特點:

  • 函數可以作為變數、參數、返回值和數據類型。
  • 基於表達式來替代方法的調用
  • 函數無狀態,可以併發和獨立使用
  • 函數無副作用,不會修改外部的變數
  • 函數結果確定性;同樣的輸入,必然會有同樣的結果。

下麵jdk1.8裡面對函數式編程的定義。只是一個  FunctionalInterface 介面特別的簡單。

1 @Documented
2 @Retention(RetentionPolicy.RUNTIME)
3 @Target(ElementType.TYPE)
4 public @interface FunctionalInterface {}

這個函數式介面有幾點以下的限制:

  • 唯一的抽象方法,有且僅有一個 (即所有的函數式介面,有且只能有一個抽象方法)
  • 加上標註,則會觸發JavaCompiler的檢查。對於符合函數介面的介面,加不加都無關緊要,但是加上則會提供一層編譯檢查的保障。如果不符合,則會報錯。 
  • 不能被覆蓋之後,再聲明為抽象方法,則不算抽象方法。例如介面實現了Object中的方法。 
  • 可用於lambda類型的使用方式 

 

二. Java8新增函數式介面

Stream的操作是建立在函數式介面的組合之上的。Java8中新增的函數式介面都在java.util.function包下。這些函數式介面可以有多種分類方式。

2.1 Function

Function是從T到R的一元映射函數。將參數T傳遞給一個函數,返回R。即R = Function(T)

Function最常用的應該是  <R> Stream<R> map(Function<? super T, ? extends R> mapper);

比如List<Person> person裡面有age,name.... 我傳入age,他就會返回age的集合給我。

 1 @FunctionalInterface
 2 public interface Function<T, R> {
 3 
 4     R apply(T t);
 5 
 6     default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
 7         Objects.requireNonNull(before);
 8         return (V v) -> apply(before.apply(v));
 9     }
10 
11     default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
12         Objects.requireNonNull(after);
13         return (T t) -> after.apply(apply(t));
14     }
15 
16     static <T> Function<T, T> identity() {
17         return t -> t;
18     }
19 }

 

2.2 Predicate

Predicate是一個謂詞函數,主要作為一個謂詞演算推導真假值存在,返回布爾值的函數。Predicate等價於一個Function的boolean型返回值的子集。

predicate最常用的莫過於  Stream<T> filter(Predicate<? super T> predicate);  

比如我要過濾年齡 > 18 的人,我傳入age,判斷是否為true。為true則保留,false丟棄。

 1 @FunctionalInterface
 2 public interface Predicate<T> {
 3 
 4     boolean test(T t);
 5 
 6     default Predicate<T> and(Predicate<? super T> other) {
 7         Objects.requireNonNull(other);
 8         return (t) -> test(t) && other.test(t);
 9     }
10 
11     default Predicate<T> negate() {
12         return (t) -> !test(t);
13     }
14 
15     default Predicate<T> or(Predicate<? super T> other) {
16         Objects.requireNonNull(other);
17         return (t) -> test(t) || other.test(t);
18     }
19 
20     static <T> Predicate<T> isEqual(Object targetRef) {
21         return (null == targetRef)
22                 ? Objects::isNull
23                 : object -> targetRef.equals(object);
24     }
25 }

 

2.3 Consumer

Consumer是從T到void的一元函數,接受一個入參但不返回任何結果的操作。

Consumer最常用的肯定是   default void forEach(Consumer<? super T> action) {}

這是一段forEach迴圈的代碼,傳入實現的方法,並不返回任何值。只是迴圈。

 1 @FunctionalInterface
 2 public interface Consumer<T> {
 3 
 4     void accept(T t);
 5 
 6     default Consumer<T> andThen(Consumer<? super T> after) {
 7         Objects.requireNonNull(after);
 8         return (T t) -> { accept(t); after.accept(t); };
 9     }
10 }

 

三. Lambda表達式

3.1 基本語法

Lambda 的基本結構為 (arguments) -> body,有如下幾種情況:

  • 參數類型可推導時,不需要指定類型,如 (a) -> System.out.println(a)
  • 當只有一個參數且類型可推導時,不強制寫 (), 如 a -> System.out.println(a)
  • 參數指定類型時,必須有括弧,如 (int a) -> System.out.println(a)
  • 參數可以為空,如 () -> System.out.println(“hello”)
  • body 需要用 {} 包含語句,當只有一條語句時 {} 可省略

 

3.2 Lambda原理

比如如下代碼:

 1 List<Integer> list = new ArrayList<>();
 2 
 3 list.stream().filter((x) -> x >= 18)
 4 
 5 Stream<T> filter(Predicate<? super T> predicate);
 6 
 7 @FunctionalInterface
 8 public interface Predicate<T> {
 9 
10     boolean test(T t);
11 
12 }

比如List裡面存個個人的年齡,現在篩選出年齡大於等於18的人。

此時我們就可以用  list.stream().filter((x) -> x >= 18)   這就是一個典型的lambda表達式

(x) -> x >= 18 傳給  Predicate 函數式介面。

原理其實是:

JVM幫我們動態生成了一個內部類,然後這個內部類實現了 Predicate 這個函數式介面。

重寫了裡面的test方法。生成的類似如下:

1 static final class Main$$Lambda$1 implements Predicate<Integer> {
2     private Main$$Lambda$1() {
3     }
4 
5     @Override
6     public boolean test(Integer x) {
7         return x >= 18;
8     }
9 }

 

3.3 Lambda用法

 1 public class Main {
 2     public static void main(String[] args) {
 3 
 4         List<Integer> list = new ArrayList<>();
 5         list.add(40);
 6         list.add(50);
 7         list.add(20);
 8         list.add(30);
 9         List<Integer> collect = list.stream().filter(x -> x >= 30)
10                 .map((x) -> x + 10).sorted((x, y) -> -x.compareTo(y))
11                 .collect(Collectors.toList());
12         System.out.println(collect);
13     }
14 }

 

這個一段很典型的Lambda + Stream的用法。

  • list.stream()獲取list的stream的流
  • filter篩選出年齡大於30的人 (裡面是一個Predicate介面,返回真假)
  • map做一個function映射
  • sort排序,裡面是compartor

 

四. 總結

Lambda 表達式可以減少很多代碼,能提高生產力。但也要理解其原理。比如3.3中的代碼,為什麼filter裡面是斷言表達式,map裡面是function表達式。

這都要從lambda的原理入手,也就是JVM動態生成一個內部類,並繼承其中的抽象方法。

本次主要介紹了Java函數式編程的原理以及應用,主要從Stream和lambda入手。通過一些簡單的概念,以及代碼,更好的理解Java的函數式編程。

掌握Java的函數式編程,對平時我們開發代碼,看其他人的代碼,都有很大的幫助。

且行且珍惜,加油!

 


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

-Advertisement-
Play Games
更多相關文章
  • 對於剛入門的springboot的新手來說,學的過程中碰到的一些問題記錄下。 1. 首先,配置好Maven環境及本地倉庫 之後進入Maven安裝目錄conf文件夾下的settings.xml配置文件,用Notepadd++打開文件。 配置本地倉庫指向自己創建的本地倉庫,如圖 把jdk版本固定為1.8 ...
  • 對Series的理解也源於對其相關的代碼操作,本次僅貼一些代碼來加深理解以及記憶 ...
  • 第十二章 Django框架 12.1 伺服器程式和應用程式 伺服器程式負責對socket伺服器進行封裝,併在請求到來時,對請求的各種數據進行整理。應用程式則負責具體的邏輯處理。為了方便應用程式的開發,就出現了眾多的Web框架,例如:Django、Flask、web.py 等。不同的框架有不同的開發方 ...
  • 一、使用JSONObject來解析JSON數據官方提供的,所以不需要導入第三方jar包;直接上代碼,如下 步驟解讀: 定義一個JSON數組,用於將伺服器返回的數據傳入到一個JSONArray對象中; 然後迴圈遍歷這個JSONArray,從中取出每一個元素(JSONObject對象),接下來只需調用g ...
  • String轉成jsonObject JsonObject json = JsonObject.fromObject(String str) String轉成JsonArray JsonArray jsonArray = JsonArray.fromObject(String str) 在開發過程中 ...
  • Spring Security 解析(二) —— 認證過程   在學習Spring Cloud 時,遇到了授權服務oauth 相關內容時,總是一知半解,因此決定先把Spring Security 、Spring Security Oauth2 等許可權、認證相關的內容、原理及設計學 ...
  • 一、概述 1. 什麼是代理 我們大家都知道微商代理,簡單地說就是代替廠家賣商品,廠家“委托”代理為其銷售商品。關於微商代理,首先我們從他們那裡買東西時通常不知道背後的廠家究竟是誰,也就是說,“委托者”對我們來說是不可見的;其次,微商代理主要以朋友圈的人為目標客戶,這就相當於為廠家做了一次對客戶群體的 ...
  • 本篇概述==> 數組(依舊只敘述與C++不同的地方,或者一些應用方法例子,畢竟語言是共通的,多了也是廢話.jpg) 一,如果創建一維數組,二維數組,以及 N維數組(以此類推)呢 二,數組的一些方法(排序,複製.......) 1. 增強型 For 迴圈 用來取值,不能修改數組裡的值,相當於 普通 f ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...