淺談函數式編程

来源:https://www.cnblogs.com/roushi17/archive/2019/12/04/fp_style.html
-Advertisement-
Play Games

函數式編程(Functional Programming)是一種編程風格,它是相對於指令式編程風格而言的,常見的面向對象編程就是指令式編程風格。 指令式編程是面向電腦硬體的抽象,有變數(對應著存儲單元),賦值語句(獲取、存儲指令),表達式(記憶體引用和算術運算)和控制語句(跳轉語句)。 而函數式編程 ...


函數式編程(Functional Programming)是一種編程風格,它是相對於指令式編程風格而言的,常見的面向對象編程就是指令式編程風格。

指令式編程是面向電腦硬體的抽象,有變數(對應著存儲單元),賦值語句(獲取、存儲指令),表達式(記憶體引用和算術運算)和控制語句(跳轉語句)。

而函數式編程是面向數學的抽象,將計算描述為一種表達式求值。這裡的函數實際就是數學中的函數,即自變數到因變數的映射。也就是說,一個函數的值僅決定於函數參數的值,不依賴其他狀態。

函數式編程是一種抽象程度很高的編程範式,純粹的函數式編程語言編寫的函數沒有變數,因此,任意一個函數,只要輸入是確定的,輸出就是確定的,這種純函數我們稱之為沒有副作用。而允許使用變數的程式設計語言,由於函數內部的變數狀態不確定,同樣的輸入,可能得到不同的輸出,因此,這種函數是有副作用的。

在函數式語言當中,函數作為一等公民,可以在任何地方定義,在函數內或函數外,可以作為函數的參數或返回值,可以對函數進行組合,也可以將函數賦值給變數。嚴格意義上的函數式編程意味著不適用可變的變數,賦值,迴圈和其他命令式控制結構進行編程。

函數式編程風格帶來的好處是:

  1. 函數式編程使用不可變對象作為變數,不會修改變數的值,而是返回一個新的值,如此這樣,更容易理清頭緒,使得單元測試和調試更加容易;
  2. 可以很自由地傳遞不可變對象,但對於可變對象來說,傳遞給其他代碼之前,需要先建造一個以防萬一的副本;
  3. 一旦不可變對象完成構造以後,就不會有線程因為併發訪問而破壞對象內部狀態,因為根本沒有線程可以改變不可變對象的狀態;
  4. 不可變對象讓哈希表鍵值更安全,所以哈希表鍵要求必須是不可變對象,否則使用可變對象,如果對象狀態發生變化,那麼在哈希表中就找不到這個對象了;

具體到編程語言,Scala(靜態語言)和Python(動態語言)都能比較的支持函數式編程風格,但是它們都不是純函數式的,也就是說它們同時支持指令式風格和函數式風格。而Java基本是指令式風格,但自從Java8引入lambda表達式以後也開始部分支持函數式風格。函數式編程最典型的是諸如map, flatMap, reduce, filter等函數,它們的特點是支持某個函數作為上面這些函數的參數。

下麵分別以Java、Scala和Python舉例函數式編程,其中Java對函數式編程只是間接的支持(通過函數式介面),支持度比較有限,而Scala和Python就對函數式編程支持的比較好。

Java函數式編程舉例:

 1 package lxy.java.fp;
 2 
 3 import java.util.*;
 4 import java.util.function.*;
 5 
 6 
 7 public class FPDemo {
 8     //定義泛型方法,用以根據第二個參數指定的條件從第一個參數指定的集合中過濾部分元素,並返回過濾後的結果。這裡的第二個參數是一個函數式介面。
 9     public static <T> List <T> filter(List <T> list, Predicate <T> p) {
10         List <T> results = new ArrayList <>();
11         for (T s : list) {
12             if (p.test(s)) {
13                 results.add(s);
14             }
15         }
16         return results;
17     }
18 
19     public static void main(String[] args) {
20         List <String> myList = Arrays.asList("Hello", "Java", "Python", "Scala");
21 
22         //通過匿名類的方式
23         List <String> results = filter(myList, new Predicate <String>() {
24             public boolean test(String t) {
25                 return t.length() >= 5;
26             }
27         });
28         System.out.println("through anonymous class:");
29         System.out.println("strings with length more than 5:");
30         for (String result : results) {
31             System.out.println(result);
32         }
33 
34         //行為參數化,通過匿名函數(即lambda表達式)方式,
35         System.out.println("through lambda expression:");
36         System.out.println("strings with length more than 5:");
37         List <String> results2 = filter(myList, s -> s.length() >= 5);
38         results2.forEach(s -> System.out.println(s));
39 
40         //很容易地將過濾條件由字元串長度大於等於5改為字元串以字母a結尾,
41         // 這就是行為參數化,即將具體的邏輯(即行為或者函數)參數化,使得filter函數更加抽象,提高了代碼復用度
42         //否則需要寫2個filter函數,一個過濾出長度大於等於5的字元串,另一個過濾出以字元a結尾的字元串
43         System.out.println("strings ends with character 'a'");
44         List <String> results3 = filter(myList, s -> s.endsWith("a"));
45         results3.forEach(System.out::println);
46 
47         //Java流中的filter函數和map函數,註意這裡的filter是Java庫函數,和前面自定義的filter函數不一樣
48         System.out.println("strings with length more than 5 and its length:");
49         myList.parallelStream().filter(s -> s.length() >= 5).map(s -> "(" + s + ", " + s.length() + ")")
50                 .forEach(System.out::println);
51 
52         //高階函數
53         Function <Integer, Integer> f = (Integer x) -> x + 1;
54         Function <Integer, Integer> g = (Integer x) -> x * x;
55         Function <Integer, Integer> h = f.andThen(g);
56         Function <Integer, Integer> r = f.compose(g);
57 
58         System.out.println("higher-order function:");
59         System.out.println("h(2)=g(f(2))=" + h.apply(2));
60         System.out.println("r(2)=f(g(2))=" + r.apply(2));
61 
62     }
63 }

 

Scala函數式編程舉例:

 1 package lxy.scala.fp
 2 
 3 object FPDemo {
 4     //定義泛型方法,用以根據第二個參數指定的條件從第一個參數指定的列表中過濾部分元素,並返回過濾後的結果,結果類型仍然是List。
 5     //這裡的第二個參數是一個函數, 該函數輸入參數類型為T,返回值類型為Boolean
 6     def filter[T](list: List[T], f: T => Boolean) =
 7         for {e <- list if f(e) == true} yield e
 8 
 9     //高階函數,該函數定義為g(f(x)),其中函數f和g都是作為參數在調用該高階函數時指定的
10     def highOrderFunction1(x: Int, f: Int => Int, g: Int => Int) = g(f(x))
11 
12     //定義嵌套函數,針對每個參數,外層函數都會返回一個函數,即內層函數
13     //這裡factor是自由變數,number是綁定變數。
14     //閉包是一個函數,返回值依賴於聲明在函數外部的一個或多個變數
15     //函數multiplier返回的函數就是閉包,factor就是外部的變數,也叫自由變數,number是綁定變數(形式參數)
16     def multiplier(factor: Int): Int => Int = {
17         def multiplyByFactor(number: Int) = factor * number
18 
19         return multiplyByFactor
20     }
21 
22     //柯里化函數,類型是(Int)(Int) => Int
23     def multiplier2(factor: Int)(number: Int) = factor * number
24 
25     //這個函數實際跟multiplier效果是一樣的,每傳入一個參數factor,都會返回一個函數
26     //閉包是一個函數,返回值依賴於聲明在函數外部的一個或多個變數
27     //函數multiplier3返回的函數就是閉包,factor就是外部的變數,也叫自由變數,number是綁定變數(形式參數)
28     def multiplier3(factor: Int) = multiplier2(factor) _
29 
30     def main(args: Array[String]): Unit = {
31         val myList = List("Hello", "Java", "Python", "Scala")
32         println("strings with length more than 5")
33         filter(myList, (s: String) => s.length >= 5).foreach(s => println(s))
34 
35         println("strings ends with character 'a'")
36         filter(myList, (s: String) => s.endsWith("a")).foreach(println)
37 
38         println("strings with length more than 5 and its length:")
39         //這裡的filter不是自定義函數filter,而是庫函數,返回長度大於等於5的字元串以及對應的長度
40         myList.filter(_.length >= 5).map(s => (s, s.length)).foreach(println)
41 
42 
43         val result = highOrderFunction1(2, (x: Int) => x + 1, (x: Int) => x * x)
44         println("f(x)=x+1, g(x)=x*x, g(f(2))=" + result)
45 
46 
47         println("multiplier(2)=" + multiplier(2))
48         println("multiplier(2)(3)=" + multiplier(2)(3))
49         val double = multiplier(2)
50         println("double(3)=" + double(3))
51 
52         println("multiplier3(3)=" + multiplier3(3))
53         println("multiplier3(3)(4)=" + multiplier3(3)(4))
54         val triple = multiplier3(3)
55         println("triple(4)=" + triple(4))
56 
57     }
58 }

 

Python函數式編程舉例:

 1 #!/usr/bin/env python3
 2 # -*- coding: utf-8 -*-
 3 
 4 if __name__ == "__main__":
 5     """
 6         Usage: ./fp_demo.py
 7     """
 8 
 9     # 定義函數filter,第一個參數是列表,第二個參數是函數
10     def filter(list, f):
11         return [e for e in list if f(e) == True]
12 
13     # python中沒有foreach函數,自己定義一個,其中第一個參數是函數,第二個參數是迭代器(列表)
14     def foreach(f, iterator):
15         for item in iterator:
16             f(item)
17 
18     myList = ["Hello", "Java", "Python", "Scala"]
19     resultList = filter(myList, lambda e: len(e) >= 5)
20     print("strings with length more than 5", sep="\n")
21     foreach(lambda e: print(e, sep="\n"), resultList)
22 
23     resultList2 = filter(myList, lambda e: e.endswith("a"))
24     print("strings ends with character 'a'", sep="\n")
25     foreach(lambda e: print(e, sep="\n"), resultList2)
26 
27     # 這裡的map函數是python內置的
28     print("strings with length more than 5 and its length:", sep="\n")
29     foreach(lambda e: print(e, sep="\n"), map(lambda e: (e, len(e)), filter(myList, lambda e: len(e) >= 5)))
30 
31     # 高階函數,該函數定義為g(f(x)),其中函數f和g都是作為參數在調用該高階函數時指定的
32     def highOrderFunction1(x, f, g):
33         return g(f(x))
34 
35     # 該函數在下麵對highOrderFunction1函數的調用中被當做參數傳入
36     def f(x):
37         return x + 1
38 
39     # 第二個參數傳入上面定義的f函數,作為第三個參數的的函數採用的是匿名函數
40     result = highOrderFunction1(2, f, g=lambda x: x * x)
41     print("f(x)=x+1, g(x)=x*x, g(f(2))=%d" % result, sep="\n")
42 
43     # 定義嵌套函數,針對每個參數,外層函數都會返回一個函數,即內層函數
44     # 這裡factor是自由變數,number是綁定變數。
45     # 閉包是一個函數,返回值依賴於聲明在函數外部的一個或多個變數
46     # 函數multiplier返回的函數就是閉包,factor就是外部的變數,也叫自由變數,number是綁定變數(形式參數)
47     def multiplier(factor):
48         def multiplyByFactor(number):
49             return factor * number
50         return multiplyByFactor
51 
52 
53     print("multiplier(2)(3)=%s" % multiplier(2)(3))
54     # double是一個函數,它將輸入參數乘以2倍以後返回
55     double = multiplier(2)
56     print("double(3)=%s" % double(3))
57 
58     print("multiplier(3)(4)=%s" % multiplier(3)(4))
59     # triple是一個函數,它將輸入參數乘以3倍以後返回
60     triple = multiplier(3)
61     print("triple(4)=%s" % triple(4))
62 
63     # 第三個參數f是函數
64     def add(x, y, f):
65         return f(x) + f(y)
66 
67     print("2 ^ 2 + 3 ^2 = %s" % add(2, 3, lambda x: x * x))
68     print("(2 + 1) + (3 + 1) = %s" % add(2, 3, f))

 

最後來看一個數學題目,已知a<=b, ab都是整數,求下麵三個公式的值。

 可以看到這三個公式分別是求a~b的和,a~b的平方和,a~b各自的階乘的和。可以看到三個公式是類似的,都是求和,只不過分別是對自身求和,對自身的平方求和以及對自身的階乘求和,也就是說這裡有3個計算邏輯,需要對這3個計算邏輯計算出來的數求和。如果是指令式編程風格,就只能寫三個函數來解決問題。但是如果採用函數式編程風格,就可以只寫一個通用的求和函數來解決該問題,因為可以將這3個計算邏輯(函數)作為參數傳給之前的通用求和函數。下麵分別用Java,ScalaPython來解決該問題。

Java代碼

 1 package lxy.java.fp;
 2 
 3 import java.util.function.*;
 4 
 5 
 6 public class Sum4Integers {
 7     //通用求和函數,其中f是一個函數式介面,它接收一個整型參數並返回一個整型數值
 8     //採用遞歸計算方法
 9     private static int sum(Function<Integer, Integer> f, int a, int b) {
10         if (a > b)
11             return 0;
12         else
13             return f.apply(a) + sum(f, a + 1, b);
14     }
15 
16     //求階乘函數,採用遞歸演算法,較複雜,不能作為匿名函數傳入上面的通用求和參數,因此需要預先定義
17     private static int factor(int x) {
18         if (x == 0)
19             return 1;
20         else
21             return x * factor(x - 1);
22     }
23 
24     //求公式一函數,即求a~b的和
25     //作為參數的函數就是返回變數自身,較簡單,採用匿名函數
26     static int sumInts(int a, int b) {
27         return sum(x -> x, a, b);
28     }
29 
30     //求公式二函數,即求a^2 ~ b^2的和
31     //作為參數的函數就是返回變數的平方,較簡單,採用匿名函數
32     static int sumSquares(int a, int b) {
33         return sum(x -> x * x, a, b);
34     }
35 
36     //求公式三函數,即求a! ~ b!的和
37     //作為參數的函數就是求變數的階乘,較複雜(本身是遞歸函數),採用定義好的函數factor
38     static int sumFactors(int a, int b) {
39         return sum(Sum4Integers::factor, a, b);
40     }
41 
42     public static void main(String[] args) {
43         System.out.println("1+2+3+4+5=" + sumInts(1, 5));
44         System.out.println("1^2+2^2+3^2+4^2+5^2=" + sumSquares(1, 5));
45         System.out.println("1!+2!+3!+4!+5!=" + sumFactors(1, 5));
46     }
47 
48 }

 

Scala代碼

 1 package lxy.scala.fp
 2 
 3 
 4 object Sum4Integers {
 5     //通用求和函數
 6     //採用遞歸計算方法
 7     private def sum(f: Int => Int, a: Int, b: Int): Int = if (a > b) 0 else f(a) + sum(f, a + 1, b)
 8 
 9     //求階乘函數,採用遞歸演算法,較複雜,不能作為匿名函數傳入上面的通用求和參數,因此需要預先定義
10     private def factor(x: Int): Int = if (x == 0) 1 else x * factor(x - 1)
11 
12     //求公式一函數,即求a~b的和
13     //作為參數的函數就是返回變數自身,較簡單,採用匿名函數
14     def sumInts(a: Int, b: Int) = sum(x => x, a, b)
15 
16     //求公式二函數,即求a^2 ~ b^2的和
17     //作為參數的函數就是返回變數的平方,較簡單,採用匿名函數
18     def sumSquares(a: Int, b: Int) = sum(x => x * x, a, b)
19 
20     //求公式三函數,即求a! ~ b!的和
21     //作為參數的函數就是求變數的階乘,較複雜(本身是遞歸函數),採用定義好的函數factor
22     def sumFactors(a: Int, b: Int) = sum(factor, a, b)
23 
24     def main(args: Array[String]): Unit = {
25         println("1+2+3+4+5=" + sumInts(1, 5))
26         println("1^2+2^2+3^2+4^2+5^2=" + sumSquares(1, 5))
27         println("1!+2!+3!+4!+5!=" + sumFactors(1, 5))
28     }
29 
30 }

 

Python代碼

 1 #!/usr/bin/env python3
 2 # -*- coding: utf-8 -*-
 3 
 4 if __name__ == "__main__":
 5     """
 6         Usage: ./sum_integers.py
 7     """
 8 
 9     # 通用求和函數,採用遞歸計算方法
10     def sum(f, a, b):
11         if a > b:
12             return 0
13         else:
14             return f(a) + sum(f, a + 1, b)
15 
16     # 求階乘函數,採用遞歸演算法,較複雜,不能作為匿名函數傳入上面的通用求和參數,因此需要預先定義
17     def factor(x):
18         if x == 0:
19             return 1
20         else:
21             return x * factor(x - 1)
22 
23     # 求公式一函數,即求a~b的和
24     # 作為sum函數的第一個參數的函數就是返回變數自身,較簡單,採用匿名函數
25     def sumInts(a, b):
26         return sum(lambda x: x, a, b)
27 
28     # 求公式二函數,即求a^2 ~ b^2的和
29     # 作為sum函數的第一個參數的函數就是返回變數的平方,較簡單,採用匿名函數
30     def sumSquares(a, b):
31         return sum(lambda x: x * x, a, b)
32 
33     # 求公式三函數,即求a! ~ b!的和
34     # 作為sum函數的第一個參數的函數就是求變數的階乘,較複雜(本身是遞歸函數),採用定義好的函數factor
35     def sumFactors(a, b):
36         return sum(factor, a, b)
37 
38     print("1+2+3+4+5=%d" % sumInts(1, 5))
39     print("1^2+2^2+3^2+4^2+5^2=%d" % sumSquares(1, 5))
40     print("1!+2!+3!+4!+5!=%d" % sumFactors(1, 5))

 

從上面解決同一個問題的代碼量比較來看,Scala和Python比較短,而Java比較長,而且Java對函數式編程的支持目前還比較有限,因此函數式編程建議採用Scala或者Python。

 

 


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

-Advertisement-
Play Games
更多相關文章
  • 1. 按系列羅列Linux的發行版,並描述不同發行版之間的聯繫與區別 2. 安裝Centos7.6操作系統,創建一個自己名字的用戶名,並可以正常登錄,將主要步驟截圖。 3. 配置環境變數,實現執行history的時候可以看到執行命令的時間。 4. 總結Linux哲學思想。 5. 總結Linux常用命 ...
  • 用了很久的Window,心血來潮想換個系統,於是就開始踩坑Linux之路。 系統為deepin 首先基本的 設置root密碼 $:sudo passwd root [sudo] password for you: > 輸入密碼(當前用戶密碼) Enter new UNIX password: > 設 ...
  • 觸發器trigger 觸發器我們也可以認為是存儲過程,是一種特殊的存儲過程。 存儲過程:有輸入參數和輸出參數,定義之後需要調用 觸發器:沒有輸入參數和輸出參數,定義之後無需調用,在適當的時候會自動執行。 適當的時候:觸發器與表相關,當我們對這個相關的表中的數據進行DDL(數據的添加、修改、刪除)操作 ...
  • --顯示前條數據 select top(4) * from students; --pageSize:每頁顯示的條數 --pageNow:當前頁 select top(pageSize) * from students where sno not in (select top(pageSize*(p ...
  • --部門表 create table dept( deptno int primary key,--部門編號 dname nvarchar(30),--部門名 loc nvarchar(30)--地址 ); --雇員表 create table emp( empno int primary key, ...
  • 基本查詢: 實例表 1 示例表 2 --部門表 3 4 create table dept( 5 6 deptno int primary key,--部門編號 7 8 dname nvarchar(30),--部門名 9 10 loc nvarchar(30)--地址 11 12 ); 13 14 ...
  • 表: 學生(*學號,姓名,性別,年齡,專業) create table student( sno char(13) primary key, sname varchar(20) not null, ssex char(2), sage smallint, sdept varchar(30) ); 課 ...
  • 很多用於當SQL Server2017 安裝完成後開始菜單找不到啟動項無法啟動SQL Server2017 其實你只需要安裝一下SSMS-Setup-CHS就可以了 安裝完成之後就有了 SSMS-Setup-CHS 下載鏈接: 鏈接:https://pan.baidu.com/s/18EcH16Ok ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...