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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...