国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

Java下一代: 沒有繼承性的擴展,第 1 部分

開發 后端
Groovy、Scala 和 Clojure 提供了許多擴展機制,但繼承幾乎是 Java™ 語言的惟一選擇。這一期將介紹類別類、ExpandoMetaClass、隱式轉換和協議,借助它們來使用 Java 下一代語言擴展 Java 類。

Java 語言的設計有目的地進行了一定的刪減,以避免前代產品中已發現的一些問題。例如,Java 語言的設計人員感覺 C++ 中的多重繼承性帶來了太多復雜性,所以它們選擇不包含該特性。事實上,他們在該語言中很少構建擴展性選項,僅依靠單一繼承和接口。

其他語言(包括 Java 下一代語言)存在巨大的擴展潛力。在本期和接下來的兩期文章中,我將探索擴展 Java 類而不涉及繼承性的途徑。在本文中,您會了解如何向現有類添加方法,無論是直接還是通過語法糖 (syntactic sugar)。

表達式問題

表達式問題是最近的計算機科學歷史上的一個眾所周知的觀察結果,首創于貝爾實驗室的 Philip Wadler 的一篇未發表的論文(參見 參考資料)。(Stuart Sierra 在其 developerWorks 文章 “通過 Clojure 1.2 解決表達式問題” 中出色地解釋了它。在這篇文章中,Wadler 說道:

表達式問題是老問題的新名字。我們的目標是通過案例定義數據類型,在這里,在不重新編譯現有代碼的情況下,您可以將新的案例添加到數據類型和數據類型的新函數中,同時保留靜態類型安全(例如,沒有轉換)。

換句話說,您如何向一個分層結構中的類添加功能,而不求助于類型轉換或 if 語句?

我們將通過一個簡單的例子來表明表達式問題在真實世界中的表現形式。假設您公司始終假設應用程序中的長度單位為,沒有在您的類中為任何其他長度單位構建任何功能。但是,有一天,您公司與一家競爭對手合并了,而這個競爭對手始終假設長度單位為英尺

解決該問題的一種方法是,通過使用轉換方法擴展 Integer,使兩種格式之間的切換變得無關緊要。現代語言提供了多種解決方案來實現此目的;在本期中,我將重點介紹其中的 3 種:

  • 開放類
  • 包裝器類
  • 協議

Groovy 的類別和 ExpandoMetaClass

Groovy 包含兩種使用開放類 擴展現有的類的不同方式,“重新開放” 一個類定義來實現更改(例如添加、更改或刪除方法)的能力。

類別類

類別類(一種借鑒自 Objective-C 的概念)是包含靜態方法的常規類。每個方法至少接受一個參數,該參數表示方法擴充的類型。如果希望向 Integer 添加方法,例如我需要接受該類型作為第一個參數的靜態方法,如清單 1 所示:

清單 1. Groovy 的類別類

  1. class IntegerConv { 
  2.   static Double getAsMeters(Integer self) { 
  3.     self * 0.30480 
  4.   } 
  5.  
  6.   static Double getAsFeet(Integer self) { 
  7.     self * 3.2808 
  8.   } 

清單 1 中的 IntegerConv 類包含兩個擴充方法,每個擴充方法都接受一個名為 self(一個通用的慣用名稱)的 Integer 參數。要使用這些方法,我必須將引用代碼包裝在一個 use 代碼塊中,如清單 2 所示:

清單 2. 使用類別類

  1. @Test void test_conversion_with_category() { 
  2.   use(IntegerConv) { 
  3.     assertEquals(1 * 3.28081.asFeet, 0.1
  4.     assertEquals(1 * 0.304801.asMeters, 0.1
  5.   } 

清單 2 中有兩個特別有趣的地方。首先,盡管 清單 1 中的擴展方法名為 getAsMeters(),但我將它稱為 1.asMeters。Groovy 圍繞 Java 中的屬性的語法糖使我能夠執行 getAsMeters() 方法,好像它是名為 asMeters 的類的一個字段一樣。如果我在擴展方法中省略了 as,對擴展方法的調用需要使用空括號,就像 1.asMeters() 中一樣。一般而言,我喜歡更干凈的屬性語法,這是編寫特定于域的語言 (DSL) 的一種常見技巧。

清單 2 中第二個需要注意的地方是對 asFeetasMeters 的調用。在 use 代碼塊中,我同等地調用新方法和內置方法。該擴展在 use 代碼塊的詞法范圍內是透明的,這很好,因為它限制了擴充(有時是一些核心)類的范圍。

ExpandoMetaClass

類別是 Groovy 添加的第一種擴展機制。但事實證明對構建 Grails(基于 Groovy 的 Web 框架)而言,Groovy 的詞法范圍限制太多了。由于不滿類別中的限制,Grails 的創建者之一 Graeme Rocher 向 Groovy 添加了另一種擴展機制:ExpandoMetaClass

ExpandoMetaClass 是一種懶惰實例化的擴展持有者,它可從任何類 “成長” 而來。清單 3 展示了如何使用 ExpandoMetaClass,為我的 Integer 類實現我的擴展:

清單 3. 使用 ExpandoMetaClass 擴展 Integer

  1. class IntegerConvTest{ 
  2.  
  3.   static { 
  4.     Integer.metaClass.getAsM { -> 
  5.       delegate * 0.30480 
  6.     } 
  7.  
  8.     Integer.metaClass.getAsFt { -> 
  9.       delegate * 3.2808 
  10.     } 
  11.   } 
  12.  
  13.   @Test void conversion_with_expando() { 
  14.     assertTrue 1.asM == 0.30480 
  15.     assertTrue 1.asFt == 3.2808 
  16.   } 

在 清單 3 中,我使用 metaClass holder 添加 asMasFt 屬性,采用與 清單 2 相同的命名約定。對 metaclass 的調用出現在測試類的一個靜態初始化器中,因為我必須確保擴充操作在遇到擴展方法之前發生。

類別類和 ExpandoMetaClass 都在內置方法之前調用擴展類方法。這使您能夠添加、更改或刪除現有方法。清單 4 給出了一個示例:

清單 4. 取代現有方法的擴展類

  1. @Test void expando_order() { 
  2.   try { 
  3.     1.decode() 
  4.   } catch(NullPointerException ex) { 
  5.     println("can't decode with no parameters"
  6.   } 
  7.   Integer.metaClass.decode { -> 
  8.     delegate * Math.PI; 
  9.   } 
  10.   assertEquals(1.decode(), Math.PI, 0.1

清單4 中的第一個 decode() 方法調用是一個內置的靜態 Groovy 方法,它設計用于更改整數編碼。正常情況下,它會接受一個參數;如果調用時沒有任何參數,它將拋出 NullPointerException。但是,當我使用自己的 decode() 方法擴充 Integer 類時,它會取代原始類。

Scala 的隱式轉換

Scala 使用包裝器類 來解決表達式問題的這個方面。要向一個類添加一個方法,可將它添加到一個幫助類中,然后提供從原始類到您的幫助器的隱式轉換。在執行轉換之后,您就可以從幫助器隱式地調用該方法,而不是從原始類調用它。清單 5 中的示例使用了這種技術:

#p#

清單 5. Scala 的隱式轉換

  1. class UnitWrapper(i: Int) { 
  2.   def asFt = { 
  3.     i * 3.2808 
  4.   } 
  5.  
  6.   def asM = { 
  7.     i * 0.30480 
  8.   } 
  9.  
  10. implicit def unitWrapper(i:Int) = new UnitWrapper(i) 
  11.  
  12. println("1 foot = " + 1.asM + " meters"); 
  13. println("1 meter = " + 1.asFt + "foot"

在清單5中,我定義了一個名為 UnitWrapper 的幫助器類,它接受一個構造函數參數和兩個方法:asFtasM。在擁有轉換值的幫助類后,我創建了一個 implicit def,實例化一個新的 UnitWrapper。要調用該方法,可以像調用原始類的一個方法那樣調用它,比如 1.asM。當 Scala 未在 Integer 類上找到 asM 方法時,它會檢查是否存在隱式轉換,從而允許將調用類轉換為一個包含目標方法的類。像 Groovy 一樣,Scala 擁有語法糖,因此我能夠省略方法調用的括號,但這是一種語言特性而不是命名約定。

Scala 中的轉換幫助器通常是 object 而不是類,但我使用了一個類,因為我希望傳遞一個值作為構造函數參數(object 不允許這么做)。

Scala 中的隱式轉換是一種擴充現有類的精妙且類型安全的方式,但不能向開放類一樣,使用這種機制更改或刪除現有方法。

Clojure 的協議

Clojure 采用了另一種方法來解決表達式問題的這個方面,那就是結合使用 extend 函數和 Clojure 協議 抽象。協議在概念上類似于一個 Java 接口:一個沒有實現的方法簽名集合。盡管 Clojure 實質上不是面向對象的,而是偏向于函數,但您可以與類進行交互(并擴展它們),并將方法映射到函數。

為了擴展數字以添加轉換,我定義了一個協議,它包含我的兩個函數(asFasM)。我可使用該協議 extend 一個現有類(比如 Number)。extend 函數接受目標類作為第一個參數,接受該協議作為第二個參數,以及一個使用函數名為鍵并使用實現(以匿名函數形式)為值的映射。清單 6 顯示了 Clojure 單位轉換:

清單 6. Clojure 的擴展協議

  1. (defprotocol UnitConversions 
  2.   (asF [this]) 
  3.   (asM [this])) 
  4.  
  5. (extend Number 
  6.   UnitConversions 
  7.   {:asF (fn [this] (* this 3.2808)) 
  8.    :asM #(* % 0.30480)}) 

我可以在 Clojure REPL(interactive read-eval-print loop,交互式讀取-重新運算-打印循環)上使用新的擴展來驗證該轉換:

  1. user=> (println "1 foot is " (asM 1" meters"
  2. 1 foot is  0.3048  meters 

在清單6中,兩個轉換函數的實現演示了匿名函數聲明的兩種語法變體。每個函數只接受一個參數(asF 函數中的 this)。單參數函數很常見,以至于 Clojure 為它們的創建提供了語法糖,如 AsM 函數中所示,其中 % 是參數占位符。

協議創建了一種將方法(以函數形式)添加到現有類中的簡單解決方案。Clojure 還包含一些有用的宏,使您能夠將一組擴展整合在一起。例如,Compojure Web 框架(參見參考資料)使用協議擴展各種類型,以便它們 “知道” 如何呈現自身。清單 7 顯示了來自 Compojure 中的 Renderable 的一段代碼:

清單 7. 通過協議擴展許多類型

  1. (defprotocol Renderable 
  2.   (render [this request] 
  3.     "Render the object into a form suitable for the given request map.")) 
  4.  
  5. (extend-protocol Renderable 
  6.   nil 
  7.   (render [_ _] nil) 
  8.   String 
  9.   (render [body _] 
  10.     (-> (response body) 
  11.         (content-type "text/html; charset=utf-8"))) 
  12.   APersistentMap 
  13.   (render [resp-map _] 
  14.     (merge (with-meta (response "") (meta resp-map)) 
  15.            resp-map)) 
  16.   IFn 
  17.   (render [func request] 
  18.     (render (func request)  
  19.   ; . . . 

在清單 7中,Renderable 協議是使用單個 render 函數來定義的,該函數接受一個值和一個請求映射作為參數。Clojure 的 extend-protocol 宏(它可用于將協議定義分組到一起)接受類型和實現對。在 Clojure 中,您可使用下劃線代替不關心的參數。在清單7 中,這個定義的可看見部分為 nilStringAPersistentMapIFn(Clojure 中的函數的核心接口)提供了呈現指令。(該框架中還包含其他許多類型,但為節省空間,清單中省略了它們。)可以看到這在實踐中非常有用:對于您可能需要呈現的所有類型,您可將語義和擴展放在一起定義。

結束語

在本期中,我介紹了表達式問題,剖析了 Java 下一代語言如何處理以下方面:現有類的干凈擴展。每種語言都使用一種不同的技術(Groovy 使用開放類,Scala 使用包裝器類,而 Clojure 實現了協議)來實現類似的結果。

但是,表達式問題比類型擴充更深刻。在下一期中,我將繼續討論使用其他協議功能、特征和 mix-in 的擴展。

原文鏈接:http://www.ibm.com/developerworks/cn/java/j-jn5/index.html

責任編輯:陳四芳 來源: ibm.com
相關推薦

2013-05-28 13:44:01

Java下一代

2013-07-27 21:28:44

2013-06-27 11:21:17

2013-05-28 14:02:22

GroovyScala語言

2018-09-27 18:47:45

AIOpsDevOps

2020-09-27 17:27:58

邊緣計算云計算技術

2022-08-08 11:50:45

AIOps網絡管理

2013-04-03 09:37:10

JavaJVM

2016-01-26 11:58:12

2020-06-02 08:05:28

智能電表蜂窩物聯網NB-IoT

2018-09-11 08:00:00

DevOpsAIOps機器學習

2024-02-26 14:46:53

移動計算人工智能5G

2025-01-03 09:24:10

模型架構論文

2020-09-16 10:28:54

邊緣計算云計算數據中心

2013-10-11 14:29:44

Java開發

2015-01-22 16:16:01

思科IT模式

2011-01-27 09:52:43

StuxnetZeus軟件攻擊

2010-09-01 17:05:04

無線網絡

2018-09-25 07:00:50

2009-04-06 08:42:18

Firefox瀏覽器
點贊
收藏

51CTO技術棧公眾號

3751色影院一区二区三区| 国产高清中文字幕在线| 91精品秘密在线观看| 久久免费国产视频| 欧洲一区精品| 精品久久国产字幕高潮| 久久一区二区精品| 91精品国产91久久久久久密臀| 色综合久久天天综线观看| 亚洲成人一区在线| 中文字幕剧情在线观看一区| 极品视频在线| 精品久久久久久综合日本欧美| wwwww在线观看免费视频| 亚洲成人动漫av| 神马久久影视大全| 亚洲天堂偷拍| 91视频婷婷| 99久久久久国产精品| 欧美日韩国产首页在线观看| 狠狠色伊人亚洲综合网站l| 蜜臀av性久久久久蜜臀aⅴ| 久久精品五月婷婷| 久久精品二区三区| 中文亚洲视频在线| 在线观看精品| 亚洲精品日韩综合观看成人91| 国产一区二区在线免费播放| 欧美国产精品一区| 成人av免费看| 欧美日本一区二区视频在线观看| 国产高清精品一区| 天堂电影一区| 亚洲国产一区二区视频| 首播影院在线观看免费观看电视| 五月综合激情日本mⅴ| 亚洲亚洲精品三区日韩精品在线视频 | 亚洲五月综合| 亚洲精品一区二区久| 日本高清不卡一区二区三区视频 | 91蝌蚪国产九色| 成人精品网站在线观看| 亚洲小说图片视频| 99精彩视频| 日韩精品亚洲专区| 5278欧美一区二区三区| а√资源新版在线天堂| 亚洲美女久久久| 一二三中文字幕在线| 欧美日韩一区二区电影| 人狥杂交一区欧美二区| 久久久人成影片一区二区三区观看 | 日本黄色一区二区| 国产传媒在线| 久久久免费观看视频| 国产精品久久天天影视| 日韩欧美电影一区二区| 精品黄色一级片| 国内精品一区二区| 99精品国产视频| 亚洲精品男人| 日韩电影大片中文字幕| 满满都是荷尔蒙韩剧在线观看| 欧美亚州韩日在线看免费版国语版| 亚洲色精品三区二区一区| 国产亚洲毛片| 日本在线精品视频| 999在线观看精品免费不卡网站| 久久久久久亚洲| av午夜在线观看| 精品久久久久久久久中文字幕| 在线播放蜜桃麻豆| 亚洲一区二区欧美| 欧产日产国产精品视频| 日韩av色综合| 激情图片小说一区| 激情视频国产| 亚洲国产精久久久久久久| 亚洲精品一级二级三级| 一区二区三区四区久久| 欧美日韩国产中字| 日本中文字幕一级片| 亚洲一区二区三区在线| 成人欧美大片| 亚洲在线观看视频| 日本一二区不卡| 91av视频在线免费观看| 高清一区二区| 韩国成人av| 国产一区欧美二区| 视频精品导航| 欧美在线看片a免费观看| 精品动漫一区二区三区| 牛牛精品一区二区| 亚洲精品女av网站| 久久久精品黄色| 视频一区二区视频| 成人黄色生活片| 91免费观看在线| 亚洲在线视频一区二区| 黄色91在线观看| 136福利精品导航| 日韩video| 在线国产欧美| 69国产精品视频| 日韩在线小视频| 午夜免费性福利| 日本精品视频一区| 日本免费在线视频观看| 国产日韩精品一区二区三区 | 妺妺窝人体色www在线小说| 亚洲人成网站在线| 久草视频视频在线播放| 美日韩黄色大片| 精产国品自在线www| 久久综合狠狠综合久久综青草| 精品国产区一区| 国产精品嫩草99a| 欧美伦理在线视频| 91av俱乐部| yy6080久久伦理一区二区| 欧美日韩美女视频| 精品自拍偷拍| 性生活免费在线观看| 欧美精品在线看| 成人av在线看| 国产精品久久久久久久久免费高清 | 亚洲影院色在线观看免费| 一区二区欧美国产| 日本大片在线观看| 国产精品日韩欧美| 亚洲免费资源在线播放| 老汉色老汉首页av亚洲| 欧美婷婷精品激情| 久久躁狠狠躁夜夜爽| 99久久久无码国产精品| 三级黄色的网站| 麻豆国产精品va在线观看不卡| 91丨九色丨国产丨porny| 国产资源一区| 狠狠操精品视频| 午夜精品久久久久久久久久久久久 | 国产丝袜不卡| 欧美色涩在线第一页| 亚洲精品乱码| 一级毛片视频在线观看| 久久国产精品 国产精品| 日韩午夜在线观看视频| 久久99深爱久久99精品| 狠狠热免费视频| 欧美专区在线观看| 日韩影院免费视频| 高潮在线视频| 人人妻人人澡人人爽欧美一区双 | 麻豆福利在线观看| 国产精品第157页| 欧美精品激情在线| 亚洲第一久久影院| 在线亚洲一区| 巨胸喷奶水www久久久| 男女无套免费网站| 国产精品美女黄网| 亚洲精品网站在线播放gif| 久久品道一品道久久精品| 飘雪影院手机免费高清版在线观看 | 国产精品揄拍一区二区| 亚洲一二三专区| 国产精品亚洲综合久久| 亚洲精品福利电影| 成人网址大全| 国产麻豆日韩| 中文字幕国产亚洲| 一区二区三区在线免费观看| 国产精品vip| 激情亚洲小说| 国产最新视频在线| 日韩在线观看a| 91麻豆国产精品| 日韩精品视频免费专区在线播放| 国产精品色哟哟网站| 国产婷婷精品| 国产午夜精品一区在线观看| 日韩欧美亚洲系列| www.av91| 成人自拍爱视频| 精品国产一区二区三区久久狼5月| 欧美黄页在线免费观看| 啦啦啦中文高清在线视频| 久久久久九九九| 欧美精品在线极品| 欧美日韩专区在线| 国产亚洲欧洲997久久综合| 欧美日韩国产亚洲一区| 亚洲一区二区av| 日韩av在线高清| 亚洲色图丝袜美腿| 久久99九九99精品| 午夜精品一区二区三区国产| 欧美日韩美女| 免费观看成人高潮|