摘要:前言這周我準備介紹一個有趣的但是很少使用的方法按照合約編程,又稱為合約編程,是一種軟件設計的方法。這些規(guī)則被稱為合約,可以比擬為商業(yè)合同中的條件和義務。通過將檢查和異常拋出指令包裝到方法中,人們可以很容易地實現(xiàn)合約式編程。
前言
這周我準備介紹一個有趣的但是很少使用的方法
按照合約編程,又稱為合約編程,是一種軟件設計的方法。它規(guī)定了軟件設計師應該為軟件組件定義正式,精確和可驗證的接口規(guī)范,將常規(guī)的抽象數(shù)據(jù)類型擴展為前置條件,后置條件和不變量。這些規(guī)則被稱為合約,可以比擬為商業(yè)合同中的條件和義務。
— Wikipedia
https://en.wikipedia.org/wiki...
本質(zhì)上它使得計算盡快的因為錯誤而失敗。如果從假設條件開始就不滿足,那么沒有必要繼續(xù)運行代碼。
讓我們使用兩個銀行之間的轉(zhuǎn)賬操作作為例子說明。以下是一些條件:
前置條件:
轉(zhuǎn)賬的數(shù)額必須大于0
不變量:
轉(zhuǎn)出的銀行賬號的余額必須為正
轉(zhuǎn)賬之后:
源銀行賬戶余額必須等于初始余額減去轉(zhuǎn)賬金額
目標銀行賬戶余額必須等于初始余額加轉(zhuǎn)移金額
簡單的實現(xiàn)可以手動實現(xiàn)前置條件后置條件:
public void transfer(Account source, Account target, BigDecimal amount) { if (amount.compareTo(BigDecimal.ZERO) <= 0) { throw new IllegalArgument("Amount transferred must be higher than zero (" + amount + ")"; } if (source.getBalance().compareTo(BigDecimal.ZERO) <= 0) { throw new IllegalArgument("Source account balance must be higher than zero (" + source.getBalance() + ")"; } source.transfer(target, amount); if (source.getBalance().compareTo(BigDecimal.ZERO) <= 0) { throw new IllegalState("Source account balance must be higher than zero (" + source.getBalance() + ")"; } // Other post-conditions... }
寫起來非常麻煩,而且很難閱讀。
檢查不變式翻譯為既檢查前提條件又檢查后置條件Java語言實現(xiàn)
你可能已經(jīng)通過assert關鍵字熟悉了前置條件和后置條件:
public void transfer(Account source, Account target, BigDecimal amount) { assert (amount.compareTo(BigDecimal.ZERO) <= 0); assert (source.getBalance().compareTo(BigDecimal.ZERO) <= 0); source.transfer(target, amount); assert (source.getBalance().compareTo(BigDecimal.ZERO) <= 0); // Other post-conditions... }
Java語言實現(xiàn)有幾個問題:
前置條件和后置條件沒有區(qū)別
需要使用-ea標記啟動
Oracle的文檔明確說明:
雖然assert構造不是一個完整的合約編程工具,但它可以幫助支持非正式的按照合約設計的編程風格。其它的Java語言實現(xiàn)
自從Java 8之后,Objects類的三個方法提供了對合約式編程的部分支持:
public static
public static
public static
最后一個方法中的Supplier參數(shù)返回錯誤信息
所有的3個方法都會在obj為null的時候拋出NullPointerException。更有意思的是,他們都會在obj不是null的時候返回該對象。從而導致了以下風格的代碼:
public void transfer(Account source, Account target, BigDecimal amount) { if (requireNonNull(amount).compareTo(BigDecimal.ZERO) <= 0) { throw new IllegalArgument("Amount transferred must be higher than zero (" + amount + ")"; } if (requireNonNull(source).getBalance().compareTo(BigDecimal.ZERO) <= 0) { throw new IllegalArgument("Source account balance must be higher than zero (" + source.getBalance() + ")"; } source.transfer(target, amount); if (requireNonNull(source).getBalance().compareTo(BigDecimal.ZERO) <= 0) { throw new IllegalState("Source account balance must be higher than zero (" + source.getBalance() + ")"; } // Other post-conditions... }
不僅功能有限,而且并不能真正提高可讀性,特別是如果添加錯誤消息參數(shù)的時候。
特定框架的實現(xiàn)Spring框架提供了Assert類并支持大量的條件驗證方法。
根據(jù)我們自己簡單的實現(xiàn),前置條件不符合會拋出IllegalArgumentException,而后置條件不符合會拋出IllegalStateException。
維基百科頁面還列出了幾個專用于按合同進行編程的框架:
OVal
Contracts for Java
Java Modeling Language
Bean Validation
valid4j
上面的框架大多數(shù)基于注解。
注解的優(yōu)點和缺點讓我們從優(yōu)點開始:注釋使條件更加明顯。
而另一方面,它們也有以下缺陷:
它們需要在編譯時或運行時進行字節(jié)碼操作
它們要么:
范圍有限(比如@email)
或者委托給一個外部的語言,該語言被配置為注釋字符串屬性,違背了類型安全
Kotlin的方法Kotlin的合約編程基于簡單的方法調(diào)用,位于Preconditions.kt文件中
require類型的方法會判斷前置條件并且在不符合時拋出IllegalArgumentException
type類型的方法會判斷后置條件并且在不符合時拋出IllegalStateException
使用Kotlin重寫后的方法如下:
fun transfer(source: Account, target: Account, amount: BigDecimal) { require(amount <= BigDecimal.ZERO) require(source.getBalance() <= BigDecimal.ZERO) source.transfer(target, amount); check(source.getBalance() <= BigDecimal.ZERO) // Other post-conditions... }總結
在通常情況下,越簡單越好。通過將檢查和異常拋出指令包裝到方法中,人們可以很容易地實現(xiàn)合約式編程。盡管在Java中沒有這種即拆即用的封裝,valid4j和Kotlin都提供了這種實現(xiàn)。
想要了解更多開發(fā)技術,面試教程以及互聯(lián)網(wǎng)公司內(nèi)推,歡迎關注我的微信公眾號!將會不定期的發(fā)放福利哦~
文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/68749.html
摘要:什么是為執(zhí)行字節(jié)碼提供一個運行環(huán)境。它的實現(xiàn)主要包含三個部分,描述實現(xiàn)規(guī)格的文檔,具體實現(xiàn)和滿足要求的計算機程序以及實例具體執(zhí)行字節(jié)碼。該類先被轉(zhuǎn)化為一組字節(jié)碼并放入文件中。字節(jié)碼校驗器通過字節(jié)碼校驗器檢查格式并找出非法代碼。 什么是Java Development Kit (JDK)? JDK通常用來開發(fā)Java應用和插件?;旧峡梢哉J為是一個軟件開發(fā)環(huán)境。JDK包含Java Run...
摘要:它們是通過來自遠程的服務器的連接發(fā)送字節(jié)碼并在本地運行,這一點令人興奮。中有一個自定義的,它不是從本地文件系統(tǒng)加載類文件,而是從遠程服務器上獲取,通過加載原始字節(jié)碼,再在中轉(zhuǎn)化為類。它將字節(jié)碼解析為運行時的數(shù)據(jù)結構,檢查其有效性等。 前言 Java ClassLoader是java運行系統(tǒng)中一個至關重要但是經(jīng)常被忽略的組件。它負責在運行時尋找并加載類文件。創(chuàng)建自定義的ClassLoad...
摘要:無需檢查的異常也是的子類。從低層拋出的需檢查異常強制要求調(diào)用方捕獲或是拋出該異常。當前執(zhí)行的線程將會停止并報告該異常。單元測試允許我在使用中查看異常,并且作為一個可以被執(zhí)行的文檔來使用。不要捕獲最高層異常繼承的異常同樣是的子類。 前言 異常處理的問題之一是知道何時以及如何去使用它。我會討論一些異常處理的最佳實踐,也會總結最近在異常處理上的一些爭論。 作為程序員,我們想要寫高質(zhì)量的能夠解...
摘要:有可能一個線程中的動作相對于另一個線程出現(xiàn)亂序。當實際輸出取決于線程交錯的結果時,這種情況被稱為競爭條件。這里的問題在于代碼塊不是原子性的,而且實例的變化對別的線程不可見。這種不能同時在多個線程上執(zhí)行的部分被稱為關鍵部分。 為什么要額外寫一篇文章來研究volatile呢?是因為這可能是并發(fā)中最令人困惑以及最被誤解的結構。我看過不少解釋volatile的博客,但是大多數(shù)要么不完整,要么難...
摘要:簡介從創(chuàng)建以來,就支持核心的并發(fā)概念如線程和鎖。這篇文章會幫助從事多線程編程的開發(fā)人員理解核心的并發(fā)概念以及如何使用它們。請求操作系統(tǒng)互斥,并讓操作系統(tǒng)調(diào)度程序處理線程停放和喚醒。 簡介 從創(chuàng)建以來,JAVA就支持核心的并發(fā)概念如線程和鎖。這篇文章會幫助從事多線程編程的JAVA開發(fā)人員理解核心的并發(fā)概念以及如何使用它們。 (博主將在其中加上自己的理解以及自己想出的例子作為補充) 概念 ...
閱讀 4320·2021-09-24 09:47
閱讀 1192·2021-09-03 10:33
閱讀 2077·2019-08-30 11:13
閱讀 1039·2019-08-30 10:49
閱讀 1762·2019-08-29 16:13
閱讀 2052·2019-08-29 11:28
閱讀 3102·2019-08-26 13:31
閱讀 3638·2019-08-23 17:14