使用ASP.NET Core 3.x 構建 RESTful API - 2. 什么是RESTful API


1. 使用ASP.NET Core 3.x 構建 RESTful API - 1.准備工作

 

什么是REST 

REST一詞最早是在2000年,由Roy Fielding在他的博士論文《Architectural Styles and the Design of Network-based Software Architecture》中提的。他在本文中創造了REST這個術語。這篇論文的地址是:https://www.ics.uci.edu/~fielding/pubs/dissertation/top.htm 

 

REST的全稱是 Representational State Transfer(狀態表述轉換)。這個詞表面看起來可能不太好理解。但其實REST就是勾畫出了這樣一幅景象,它描述了Web應用到底怎么樣設計才算是優良的。這里定義了以下三點: 

  • 一組網頁的網絡(一個虛擬狀態機); 

  • 在這些網頁上,用戶可以通過點擊鏈接來前進(狀態轉換); 

  • 點擊鏈接的結果就是下一個網頁(表示程序的下一個狀態)被傳輸到用戶那里,並渲染好給用戶使用。 

 

論文中還提到REST是一種為分布式超媒體系統所用的架構風格,也就是說,REST定義了一種架構風格來幫助創建和組織出更好的分布式系統。這里的關鍵詞是架構風格 

概括的說: 

  • REST是一種架構風格,而不是規范或標准; 

  • REST需要使用一些規范、協議或標准來實現這種架構風格; 

  • REST與協議無關。JSON並不是REST強制的,甚至HTTP都不是REST強制使用的,但這也僅僅是從理論上來看。 

 

REST背后的主要思想就是:采用RESTful架構風格進行組織的分布式系統,將在以下幾個方面得到改善: 

  • 性能。REST的通信風格應該是簡單並且高效的,采用它的系統性能應該得以提升。 

  • 組件交互的可擴展性。其實任何分布式系統都允許這種擴展性,而REST所提出的簡單交互方式更是如此。 

  • 組件的可修改性。分布式系統的分布式本質和REST提出的關注點分離,使得組件得以以最小的成本和最低的風險彼此獨立的進行修改。 

  • 可移植性。REST與技術和語言無關,所以使用任何技術都可以實現REST。 

  • 可靠性REST所提出的無狀態約束允許在系統發生故障后輕松的恢復系統。 

  • 可視性REST所提出的無狀態約束為所述請求添加了完整的狀態(一會再解釋)。 

 

從上面這個列表,我們可以看出,一個以組件為中心設計的系統非常容易出錯,如果一個組件出現了故障而不影響整個系統的穩定性,那這樣對任何系統都是極有好處的。對組件進行互聯是非常簡單的,但是需要在添加新特性或擴大縮小規模時將風險降至最低。憑借REST的可移植性,使用REST思想進行設計的系統可以為更廣泛的受眾使用。通過通用的接口,系統可以被更廣泛的開發者所使用。為了實現這些屬性和好處REST使用一組約束來幫助定義統一的接口。 

 

REST的約束 

為了定義REST架構,首先要定義出一個空無的狀態,也就是一個沒有任何約束的系統。在這里,組件之間的差異就是個迷,然后我們再一個挨一個的往里面添加約束並保證這些約束可以互不干擾、融洽相處。這些約束都定義了實現REST API的框架應該如何被構建和設計。下面就介紹一些這六個約束: 

  • 客戶端-服務器:關注點分離是這個約束的核心主題。整個Web系統是一個基於客戶端-服務端的系統,客戶端和服務端彼此獨立(獨立實現和部署等),並扮演着不同的角色。它們可以使用不同的語言、技術或平台,並可以獨自進化,只要它們都遵從Web的統一接口即可。 

  • 無狀態:無狀態表示Web服務器不被要求記住客戶端程序的狀態,因為這個原因,客戶端在發送請求的時候必須包含所有可能需要的相關信息,也就是說狀態需要被包含在請求里,同時也說明客戶端需要維護自己的狀態。由於維護狀態的工作由客戶端自己來完成了,所以服務器就節省了很多服務器資源,這樣服務器就可以為更多的客戶端服務。 

  • 統一的資源接口/界面:Web組件之間的交互就意味着客戶端、服務端以及基於網絡的中介程序都依賴於它們接口的統一性(API和API的消費者之間共用相同標准的一套接口)。Web組件可以在統一接口的四個約束條件下一致的進行互操作。這四個約束是: 

    • 資源的標識:針對RESTful Web API而言,就是指URI,只有得到這個資源標識,才有可能找到該資源並對該資源進行操作。但是從概念上來講,資源和它的表述是分開的。例如,我們通過一個URI找到了服務端的Company這個資源,但是我們得到的Company這個資源的表述和服務端的Company是不一樣的,因為我們得到的是JSON格式(大多數情況)的Company數據。同時還有媒體類型(media type)對其進行描述,例如application/json等。如果請求的是xml格式的數據,那么我們通常會得到xml格式表述的數據。所以同一個資源得到的表述也可能是不同的(例如JSON vs Xml)。 

    • 通過表述來對資源進行操縱:REST的組件對資源的操作(CRUD)是通過首先獲取該資源現有的表述或者目標表述,然后在組件之間完成從現有表述到目標表述的轉換。換句話講,當客戶端擁有資源表述的時候(包括可能的元數據),那么它就應該擁有足夠的信息來修改或者刪除服務器上的資源,前提是客戶端需要有這些權限。例如,我從服務器獲取到了Company的資源響應(包括元數據)之后,憑借這些信息客戶端就應該可以成功的刪除或修改這個Company的資源數據了。但這又是怎么實現的呢?如果服務器上的Company API支持對Company進行刪除或者修改,那么在我們獲取(GET)到這個Company資源的響應后,響應里面應該包含着刪除或者修改這個Company資源的URI,通過這些URI客戶端就可以完成相應的操作。 

    • 帶有自我描述的信息:由於REST是無狀態的(沒有會話機制),所以發送REST請求的時候,必須把所有相關的信息隨着請求一起發送到服務器端。換句話說,需要通過使用元數據或者其它方式,讓REST的請求中包含的數據必須帶有“自我描述”性的信息,以便讓對方知道如何處理該請求。 

    • 超媒體作為應用程序狀態的引擎(HATEOAS):REST架構風格中,客戶端是通過超媒體與服務器端動態提供的一個“應用網絡”來進行交互的。這里要求在首次進入REST網絡時有第一個鏈接,還要求客戶端必須具備處理超媒體內容的能力。除此之外REST對客戶端來說再無其它要求。這是書上給出的解釋。舉個例子,本文第二段中提到用戶通過點擊網頁中的鏈接來進行跳轉的時候,瀏覽器的狀態就變化了。這些鏈接就是超文本,而超媒體就是超文本的泛化。針對API來說,它就是程序狀態的引擎。換句話說,超媒體會驅動如何消費和使用API,它會告訴API消費者使用這些API能做什么,例如:能刪除這個資源嗎?能修改資源嗎?如何能創建這種資源?從哪能獲取這個資源?最終,它還允許自包含文檔的API。 

  • 多層系統:REST的解決方案適用於多層架構,這些層可以被修改,可以被添加或刪除,可以是物理的,也可以是邏輯的。每一層只可以看到和它相鄰的上一層或下一層,其它非相鄰層的結構它完全看不到。這也說明客戶端無法得知它連接的是架構最終層還是連接到了某個中間層。所以REST僅僅知道一個層,也就是對外那一層,因為這個原因,整個系統的復雜性得到了控制,因為可以對任何局部的層次進行替換,而不至於影響整個系統。 

  • 可緩存:每個響應信息必須明確的指出它是否可以被緩存。緩存響應數據可以減少客戶端感知的響應時間,提高整體的可用性和可靠性,並控制整個Web服務器的負載。客戶端也可以在實時性和響應速度之間做出選擇,以便服務器端相應的決定是從緩存還是從最終信息源哪里獲得服務響應的內容。 

  • 按需編碼(可選約束):它描述了服務器可以擴展或者定制客戶端的功能。例如如果客戶端是一個Web應用,那么服務器端可以發送一些javascript腳本給客戶端,以擴展客戶端的功能。但是這也造成了客戶端和服務器端之間的技術耦合,因為客戶端必須能都懂得服務器端發過來的代碼,所以這個約束是可選的。 

 

這些就是REST的約束,而沒有實現這些約束的Web API就不是RESTful API,所以現在見到的很多RESTful API並不是真的RESTful API,但是這也不能說明這些API就不好,只不過針對那些沒有實現的約束可能要做出一些權衡取舍,付出一些代價。 

 

Richardson 成熟度模型 

這個成熟度模型是由Leonard Richardson所提出的,這個模型是用來評價API的成熟度。它的結果分為0123共四個級別。我們一個一個看。 

  • Level 0POXPlain old xml)沼澤。它描述了API僅僅是使用HTTP協議來做遠程交互,而HTTP協議的其余部分都是瞎用的,有時用出了RPC的風格(例如SOAP, 尤其是使用WCF的時候)。例如下面這個程序都是在同一個URI上面進行讀取資源和創建資源的: 

http. 
post 
• / /h.st/mvapi
  • 換句話說,就是使用HTTP協議作為一種傳輸方式而已,沒有什么規矩可言。 

  • Level 1,資源。在這級里Level 0不同,每個資源都映射到自己的URI上了但是HTTP方法並沒有正確的使用但是還是降低了一些復雜度。例如下面這個例子使用了不同的URI,但是HTTP方法使用的都是POST: 

: / authors
  • Level 2,動詞正確使用了HTTP動詞,例如GET、POST、DELETE、PUT、PATCH等等都是按照協議的意圖正確的使用了。狀態碼也正確的使用了,例如200表示成功,201表示創建成功等等。這也符合了統一資源接口/界面這個約束。從軟件開發角度,這也去掉了不必要的變種,因為我們使用同樣的動詞來做同類的事情。例如: 

GET 
200 OZ (auth=s) 
POST 
2 OL (auZhcz)
  • Level 3,超媒體。這意味着,API支持HATEOAS(超媒體作為應用狀態的引擎, Hypermedia as the Engine of Application State,這也是統一資源接口/界面約束里面的一條例如: 

GET 
: / authors

這個GET請求的響應除了包含數據之外,還包含鏈接(超媒體),這些鏈接可以驅動應用程序的狀態。從軟件開發的角度講,就是引入了可發現性和自包含文檔 

 

根據Roy Fielding博士的描述,達到Level 3也僅僅是RESTful API的一個前提。也就是說只有你的API達到了Level 3水平之后,才可以談論你的API是不是RESTful API 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM