Scala Types 1

来源:https://www.cnblogs.com/yuanzam/archive/2019/10/29/11761509.html
-Advertisement-
Play Games

在 Scala 中所有值都有一種對應的類型 單例類型 形式: ,返回類型 / 場景1:鏈式API調用時的類型指定 由於 Scala 會將 推斷為當前類(即 ),因此無法完成鏈式調用 場景2:方法中使用 實例作為參數 Note:不可定義為 ~~def c1(obj: Foo)~~,因為 Foo 為單例 ...


在 Scala 中所有值都有一種對應的類型

單例類型

  • 形式:value.type,返回類型 value / null
  • 場景1:鏈式API調用時的類型指定

    class Super {
      def m1(t: Int) = {println(t); this}
      def m2(t: Int) = {println(t); this}
    }
    // 正常列印
    new Super().m1(1).m2(2)
    
    class Child extends Super {
      def c1(t: Int) = {println(t); this}
    }
    
    // 異常  value c1 is not a member of Super
    new Child().m1(1).c1(2)

    由於 Scala 會將 this 推斷為當前類(即 Super),因此無法完成鏈式調用

    class Super {
      // 指定返回類型為調用方的 this 
      def m1(t: Int): this.type = {println(t); this}
      def m2(t: Int): this.type = {println(t); this}
    }
    
    class Child extends Super {
      def c1(t: Int) = {println(t); this}
    }
    
    // 成功列印
    new Child().m1(1).c1(2)
  • 場景2:方法中使用 object 實例作為參數

    object Foo
    class Child extends Super {
      def c1(obj: Foo.type) = {
        if (obj == Foo) println("foo")
        this
      }
    }

    Note:不可定義為 def c1(obj: Foo),因為 Foo 為單例對象,而不是類型

類型投影

  • 形式:Outer#Inner
  • 場景:內部類使用時避免類型約束

    class Outer {
      private val inners = ArrayBuffer[Inner]()
    
      class Inner (val arg1: Int) {
        val l = ArrayBuffer[Inner]()
      }
    
      def add(a: Int) = {
        val t = new Inner(a)
        inners += t
        t
      }
    }
    
    val a = new Outer
    val b = new Outer
    
    val a1 = a.add(1)
    val b1 = b.add(1)
    a1.l += b1 // error: type mismatch;

    只需要在定義內部類時指定類型投影即可解決

    // 表示適用於任何 Outer 類的 Inner 類
    val l = ArrayBuffer[Outer#Inner]()

    如果將上述例子改用 List 來實現,並不會報錯,計算結果也會自動進行類型投射

路徑

  • 路徑中除最後一部分外,都必須是穩定狀態的,如包名、objectvalthis/super/super[S]...
  • 不能包含 var 類型

    var t = new Outer()
    //...其他操作
    val i = new t.Inner // 由於 t 可能會變更,編譯器無法確定其含義

    a.b.c.T 內部被翻譯成類型投射 a.b.c.type#T

類型別名

  • 形式: type SomeAliasName
  • 必須定義在 classobject 內部
  • 好處: 在引用類型時可以更加簡潔

    class Book {
      import scala.collection.mutable._
      // 為該類型取一個別名
      type Index = HashMap[String, Int]
    
      // 使用時不在需要重覆的定義複雜的數據類型
      val map: Index = new Index()
    }
    
    new Book().map  // scala.collection.mutable.HashMap[String,Int]

結構類型

  • 為抽象方法、欄位、類型的定義某種規範

    def appendLines(target: { def append(str: String): Any },
      lines: Iterable[String]) {
      for (l <- lines) { 
        // 此次 Scala 使用反射調用該方法
        target.append(l); 
        target.append("\n") 
      }
    }

    該方法第一個參數 target 即結構類型,表示使用任何包含該 append 方法的實例作為參數傳入。

    由於反射的代價較大,不到萬不得已不建議使用,如,有通用行為(append),卻無法共用 trait

組合類型 / 交集類型

  • 形式: T1 with T2 with T3 ...
  • 當需要提供多個特質時使用,即用於約束類型

    val image = new ArrayBuffer[java.awt.Shape with java.io.Serializable]
    val rect = new Rectangle(5, 10, 20, 30)
    image += rect // 正確,Rectangle 可序列化
    image += new Area(rect) // 錯誤 Area 不可序列化
  • 組合類型中也可使用結構類型

    Shape with Serializable { def contains(p: Point): Boolean }

中綴類型

  • 其實只是一種語法寫法,如 String Map Int 可代替 Map[String, Int]
  • 可參考數學運算中的表達方式

    type x[A, B] = (String, Int)
    // 即可使用 String x Int 來表示 (String, Int)
  • 一般中綴類型操作符都是左關聯的,除了前面提到的 : 操作符,這個是右關聯的,比如 List 的操作

中綴類型名稱可以是任意操作符,除了 *,避免與類型定義衝突


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

-Advertisement-
Play Games
更多相關文章
  • 我是一個不太愛折騰的人,因此在一個公司待久了,就不太會輕易跳槽。正因為如此,我在上家公司待了整整三年,在這裡,認識了一群可愛的人,便更不捨得離去。 但因為公司屬於傳統企業,技術上並沒有太大挑戰,個人也逐漸遇到了職業瓶頸,我也漸漸體會到了溫水煮青蛙的感覺,看似自己已經成為團隊的主程,其實與同齡人的差距 ...
  • 本文源碼: "GitHub·點這裡" || "GitEE·點這裡" 一、生活場景 1、場景描述 在電商高速發展的今天,快遞的數量十分龐大,甚至出現了快遞代理行業,簡單的說就是快遞的主人沒有時間收快遞,會指定一個快遞的代收點,比如快遞櫃,快遞驛站等,然後等有時間的時候再過去取,下麵使用代碼對這個場景進 ...
  • 一、全局變數 1. 定義在函數外面的變數是全局變數 2. 全局變數具有全局的生存期和作用域 3. 它們與任何函數無關,在任何函數內部都可以使用它們 二、全局變數初始化 1. 沒有做初始化的全局變數會得到0值 2. 指針會得到NULL值 3. 只能用編譯時刻已知的值來初始化全局變數 4. 它們的初始化 ...
  • 跟大家分享一下正式第一次使用 LaTex 的經驗,之前數學建模的時候一直想用,但沒有找到合適的軟體.前段時間,實驗室老師讓我幫忙套個 IEEE ACCESS 的模板. 嘗試過 TexPad,的確 UI 特別好看,奈何咋老報錯呢??? 然後乖乖回歸TexShop,用習慣了也不醜了. 因為老師給的文件是 ...
  • switch 是一個條件語句,用於將表達式的值與可能匹配的選項列表進行比較,並根據匹配情況執行相應的代碼塊。它可以被認為是替代多個 if else 子句的常用方式。 ...
  • 建立一棵含有n個結點的二叉樹,採用二叉鏈表存儲; 輸出前序、中序、後序、、層序遍歷該二叉樹的遍歷結果。 定義二叉樹的數據類型——二叉樹結點結構體BiNode。建立二叉鏈表可以採用擴展二叉樹的一個遍歷序列,例如前序序列,將擴展二叉樹的前序序列由鍵盤輸入,建立該二叉樹的二叉鏈表存儲。 簡單起見,本實驗假 ...
  • 在下麵這個例子中,我們分別在父類和子類中測試了 靜態代碼塊、普通代碼塊、靜態成員變數、普通成員變數、構造器、靜態內部類 。 一:代碼塊及變數測試 二:測試結果 三:測試靜態內部類 四:總結 經過我們反覆調整順序進行測試後,可得出如下結論: 1、靜態代碼塊、靜態成員變數只有第一次載入類時才會執行。 2 ...
  • 註意:eg a+2a=4 第一個a的k等於0 特例 eg -2=6y 等號右邊f要變 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...