Scala 函數式編程

来源:https://www.cnblogs.com/YuanWeiBlogger/archive/2019/08/29/11432585.html
-Advertisement-
Play Games

將函數賦值給變數 // Scala中的函數是一等公民,可以獨立定義,獨立存在,而且可以直接將函數作為值賦值給變數 // Scala的語法規定,將函數賦值給變數時,必須在函數後面加上空格和下劃線 def sayHello(name: String) { println("Hello, " + name ...


將函數賦值給變數

// Scala中的函數是一等公民,可以獨立定義,獨立存在,而且可以直接將函數作為值賦值給變數

// Scala的語法規定,將函數賦值給變數時,必須在函數後面加上空格和下劃線

 

def sayHello(name: String) { println("Hello, " + name) }

val sayHelloFunc = sayHello _

sayHelloFunc("leo")

 

 

 匿名函數

// Scala中,函數也可以不需要命名,此時函數被稱為匿名函數。// 可以直接定義函數之後,將函數賦值給某個變數;也可以將直接定義的匿名函數傳入其他函數之中

// Scala定義匿名函數的語法規則就是,(參數名: 參數類型) => 函數體

// 這種匿名函數的語法必須深刻理解和掌握,在spark的中有大量這樣的語法,如果沒有掌握,是看不懂spark源碼的

 

val sayHelloFunc = (name: String) => println("Hello, " + name)

 

 

高階函數

// Scala中,由於函數是一等公民,因此可以直接將某個函數傳入其他函數,作為參數。這個功能是極其強大的,也是Java這種面向對象的編程語言所不具備的。

// 接收其他函數作為參數的函數,也被稱作高階函數(higher-order function

val sayHelloFunc = (name: String) => println("Hello, " + name)

def greeting(func: (String) => Unit, name: String) { func(name) }

greeting(sayHelloFunc, "leo")

 

Array(1, 2, 3, 4, 5).map((num: Int) => num * num)

 

// 高階函數的另外一個功能是將函數作為返回值

def getGreetingFunc(msg: String) = (name: String) => println(msg + ", " + name)

val greetingFunc = getGreetingFunc("hello")

greetingFunc("leo")

 

高階函數的類型推斷

 

// 高階函數可以自動推斷出參數類型,而不需要寫明類型;而且對於只有一個參數的函數,還可以省去其小括弧;如果僅有的一個參數在右側的函數體內只使用一次,則還可以將接收參數省略,並且將參數用_來替代

// 諸如3 * _的這種語法,必須掌握!!spark源碼中大量使用了這種語法!

 

def greeting(func: (String) => Unit, name: String) { func(name) }

greeting((name: String) => println("Hello, " + name), "leo")

greeting((name) => println("Hello, " + name), "leo")

greeting(name => println("Hello, " + name), "leo")

 

def triple(func: (Int) => Int) = { func(3) }

triple(3 * _)

 

Scala的常用高階函數

// map: 對傳入的每個元素都進行映射,返回一個處理後的元素

Array(1, 2, 3, 4, 5).map(2 * _)

 

// foreach: 對傳入的每個元素都進行處理,但是沒有返回值

(1 to 9).map("*" * _).foreach(println _)

 

// filter: 對傳入的每個元素都進行條件判斷,如果對元素返回true,則保留該元素,否則過濾掉該元素

(1 to 20).filter(_ % 2 == 0)

 

// reduceLeft: 從左側元素開始,進行reduce操作,即先對元素1和元素2進行處理,然後將結果與元素3處理,再將結果與元素4處理,依次類推,即為reducereduce操作必須掌握!spark編程的重點!!!

// 下麵這個操作就相當於1 * 2 * 3 * 4 * 5 * 6 * 7 * 8 * 9

(1 to 9).reduceLeft( _ * _)

 

// sortWith: 對元素進行兩兩相比,進行排序

Array(3, 2, 5, 4, 10, 1).sortWith(_ < _)

 

閉包

// 閉包最簡潔的解釋:函數在變數不處於其有效作用域時,還能夠對變數進行訪問,即為閉包

 

def getGreetingFunc(msg: String) = (name: String) => println(msg + ", " + name)

val greetingFuncHello = getGreetingFunc("hello")

val greetingFuncHi = getGreetingFunc("hi")

 

// 兩次調用getGreetingFunc函數,傳入不同的msg,並創建不同的函數返回

// 然而,msg只是一個局部變數,卻在getGreetingFunc執行完之後,還可以繼續存在創建的函數之中;greetingFuncHello("leo"),調用時,值為"hello"msg被保留在了函數體內部,可以反覆的使用

// 這種變數超出了其作用域,還可以使用的情況,即為閉包

 

// Scala通過為每個函數創建對象來實現閉包,實際上對於getGreetingFunc函數創建的函數,msg是作為函數對象的變數存在的,因此每個函數才可以擁有不同的msg

 

// Scala編譯器會確保上述閉包機制

 

SAM轉換

// Java中,不支持直接將函數傳入一個方法作為參數,通常來說,唯一的辦法就是定義一個實現了某個介面的類的實例對象,該對象只有一個方法;而這些介面都只有單個的抽象方法,也就是single abstract method,簡稱為SAM

 

// 由於Scala是可以調用Java的代碼的,因此當我們調用Java的某個方法時,可能就不得不創建SAM傳遞給方法,非常麻煩;但是Scala又是支持直接傳遞函數的。此時就可以使用Scala提供的,在調用Java方法時,使用的功能,SAM轉換,即將SAM轉換為Scala函數

 

// 要使用SAM轉換,需要使用Scala提供的特性,隱式轉換

 

import javax.swing._

import java.awt.event._

 

val button = new JButton("Click")

button.addActionListener(new ActionListener {

  override def actionPerformed(event: ActionEvent) {

    println("Click Me!!!")

  }

})

 

implicit def getActionListener(actionProcessFunc: (ActionEvent) => Unit) = new ActionListener {

  override def actionPerformed(event: ActionEvent) {

    actionProcessFunc(event)

  }

}

button.addActionListener((event: ActionEvent) => println("Click Me!!!"))

 

 

Currying函數

// Curring函數,指的是,將原來接收兩個參數的一個函數,轉換為兩個函數,第一個函數接收原先的第一個參數,然後返回接收原先第二個參數的第二個函數。

// 在函數調用的過程中,就變為了兩個函數連續調用的形式

// Spark的源碼中,也有體現,所以對()()這種形式的Curring函數,必須掌握!

 

def sum(a: Int, b: Int) = a + b

sum(1, 1)

 

def sum2(a: Int) = (b: Int) => a + b

sum2(1)(1)

 

def sum3(a: Int)(b: Int) = a + b

 

Return

 

// Scala中,不需要使用return來返回函數的值,函數最後一行語句的值,就是函數的返回值。在Scala中,return用於在匿名函數中返回值給包含匿名函數的帶名函數,並作為帶名函數的返回值。

// 使用return的匿名函數,是必須給出返回類型的,否則無法通過編譯

 

def greeting(name: String) = {

  def sayHello(name: String):String = {

    return "Hello, " + name

  }

  sayHello(name)

}

 

 


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

-Advertisement-
Play Games
更多相關文章
  • 一. 什麼是IoC 當在A類中要使用B類的時候,我們一般都是採用new的方式來實例化B類,這樣一來這兩個類就有很強的依賴關係,不符合低耦合的設計思想。這時候我們可以通過一個中間容器來實例化對象,需要的時候就可以通過容器獲取一個B類的對象。這種思想就是IoC(控制反轉),其實我覺得叫控制轉移更為合適, ...
  • python提供了一個跨平臺的多進程支持——multiprocessing模塊,其包含Process類來代表一個進程對象 1、Process語法結構:(註: 傳參的時候一定使用關鍵字傳參) 2、自定義進程類:需要繼承Process類 自定義類的時候必須註意的事項: 第一,必須繼承Process類的構 ...
  • 原文:阿裡 RocketMQ 安裝與簡介 一、簡介 官方簡介: l RocketMQ是一款分散式、隊列模型的消息中間件,具有以下特點: l 能夠保證嚴格的消息順序 l 提供豐富的消息拉取模式 l 高效的訂閱者水平擴展能力 l 實時的消息訂閱機制 l 億級消息堆積能力 二、網路架構 三、特性 1. n ...
  • 原文:https://blog.csdn.net/lyly4413/article/details/80838716 1.消息中間件的發展: 第一代以ActiveMQ為代表,遵循JMS(java消息服務)規範 第二代以RabbitMQ為代表是一個有Erlang語言開發的AMQP(高級消息隊列協議)的 ...
  • 今天開始學習設計模式,藉此機會學習並整理學習筆記。 設計模式是一門不區分語言的課程,什麼樣的編程語言都可以用到設計模式。如果說java語法規則比作武功招式的話,那麼設計模式就是心法。 設計模式共有23種,常見的19種,最常用的9-10種。 設計模式分三種類型:創建型、結構型、行為型; 其中創建型包含 ...
  • Django之靜態文件,中間件,admin後臺管理;其中,靜態文件 包括靜態文件的使用,動態生成靜態文件的路徑;中間件包括 使用中間件,使用中間件防爬蟲ip案例;admin後臺管理 包括 admin的使用,列表頁選項,編輯頁選項等。 ...
  • 方法1. 函數:fill 舉例說明:應力分佈雲圖 ...
  • Scala的集合體繫結構 // Scala中的集合體系主要包括:Iterable、Seq、Set、Map。其中Iterable是所有集合trait的根trai。這個結構與Java的集合體系非常相似。 // Scala中的集合是分成可變和不可變兩類集合的,其中可變集合就是說,集合的元素可以動態修改,而 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...