91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Spring RestTemplate具體使用詳解

發布時間:2020-09-20 01:40:32 來源:腳本之家 閱讀:175 作者:KING磊 欄目:編程語言

1、什么是REST?

REST(RepresentationalState Transfer)是Roy Fielding 提出的一個描述互聯系統架構風格的名詞。REST定義了一組體系架構原則,您可以根據這些原則設計以系統資源為中心的Web 服務,包括使用不同語言編寫的客戶端如何通過 HTTP處理和傳輸資源狀態。

為什么稱為 REST?Web本質上由各種各樣的資源組成,資源由URI 唯一標識。瀏覽器(或者任何其它類似于瀏覽器的應用程序)將展示出該資源的一種表現方式,或者一種表現狀態。如果用戶在該頁面中定向到指向其它資源的鏈接,則將訪問該資源,并表現出它的狀態。這意味著客戶端應用程序隨著每個資源表現狀態的不同而發生狀態轉移,也即所謂REST。

2、REST成熟度的四個層次

第一個層次(Level0)的Web 服務只是使用 HTTP 作為傳輸方式,實際上只是遠程方法調用(RPC)的一種具體形 式。SOAP和 XML-RPC都屬于此類。

第二個層次(Level1)的Web 服務引入了資源的概念。每個資源有對應的標識符和表達。

第三個層次(Level2)的Web 服務使用不同的 HTTP 方法來進行不同的操作,并且使用HTTP 狀態碼來表示不同的結果。如 HTTPGET 方法來獲取資源,HTTPDELETE 方法來刪除資源。

第四個層次(Level3)的Web 服務使用 HATEOAS。在資源的表達中包含了鏈接信息。客戶端可以根據鏈接來發現可以執行的動作。

其中第三個層次建立了創建、讀取、更新和刪除(create,read, update, and delete,CRUD)操作與 HTTP方法之間的一對一映射。根據此映射:

(1)若要在服務器上創建資源,應該使用POST 方法。

(2)若要檢索某個資源,應該使用GET 方法。

(3)若要更改資源狀態或對其進行更新,應該使用PUT 方法。

(4)若要刪除某個資源,應該使用DELETE 方法。

3、HTTP請求的方法

(1)GET:通過請求URI得到資源
(2)POST:用于添加新的內容
(3)PUT:用于修改某個內容,若不存在則添加
(4)DELETE:刪除某個內容
(5)OPTIONS :詢問可以執行哪些方法
(6)HEAD :類似于GET, 但是不返回body信息,用于檢查對象是否存在,以及得到對象的元數據
(7)CONNECT :用于代理進行傳輸,如使用SSL
(8)TRACE:用于遠程診斷服務器

4、HTTP請求的狀態碼

(1)成功Successful2xx:此類狀態碼標識客戶端的請求被成功接收、理解并接受。常見如200(OK)、204(NoContent)。
(2)重定向Redirection3xx:這個類別的狀態碼標識用戶代理要做出進一步的動作來完成請求。常見如301(MovedPermanently)、302(MovedTemprarily)。
(3)客戶端錯誤Client Error 4xx:4xx類別的狀態碼是當客戶端象是出錯的時使用的。常見如400(BadRequest)、401(Unauthorized)、403(Forbidden)、404(NotFound)。
(4)服務器錯誤Server Error 5xx:響應狀態碼以5開頭表示服務器知道自己出錯或者沒有能力執行請求。常見如500(InternalServer Error)、502(BadGateway)、504(GatewayTimeout)。

5、RestTemplate

5.1 簡介

Spring'scentral class for synchronous client-side HTTP access.It simplifies communication with HTTPservers, and enforces RESTful principles. Ithandles HTTP connections, leaving application code to provide URLs(with possible template variables) andextract results.

簡單說就是:簡化了發起HTTP請求以及處理響應的過程,并且支持REST。為什么說簡化了呢?

來看兩種實現方式

(1)使用java.net包下的URLConnection建立連接

String result= ""; 
  BufferedReaderin = null; 
  try { 
   String urlNameString= url +"?" + param; 
   URL realUrl= new URL(urlNameString); 
   // 打開和URL之間的連接 
   URLConnectionconnection = realUrlopenConnection(); 
   // 設置通用的請求屬性 
   connectionsetRequestProperty("accept","*/*"); 
   connectionsetRequestProperty("connection","Keep-Alive"); 
   connectionsetRequestProperty("user-agent", 
     "Mozilla/0(compatible; MSIE 0; Windows NT 1;SV1)"); 
   // 建立實際的連接 
   connectionconnect(); 
   // 獲取所有響應頭字段 
   Map<String,List<String>> map = connectiongetHeaderFields(); 
   // 遍歷所有的響應頭字段 
   for(String key : mapkeySet()) { 
    Systemoutprintln(key+ "--->" + mapget(key)); 
   } 
   // 定義 BufferedReader輸入流來讀取URL的響應 
   in =new BufferedReader(newInputStreamReader( 
     connectiongetInputStream())); 
   String line; 
   while ((line = inreadLine())!= null) { 
    result += line; 
   } 
  } catch (Exception e) { 
   … 
  } 
  // 使用finally塊來關閉輸入流 
  finally{ 
   // 關閉流 
  } 

(2)使用RestTempalte

ResponseEntity<SsoUrlPrm>result = restTemplate.getForEntity(requestPathUrl,SsoUrlPrm.class); 

5.2 對外開放的接口

Spring RestTemplate具體使用詳解

(1)

DELETE delete
GET getForObject
  getForEntity
HEAD headForHeaders
OPTIONS optionsForAllow
POST postForLocation
  postForObject
PUT put
any exchange
  execute

(2)每一個小類又分三種,這三種有什么區別?* 第一種和第二種的首個參數都是用String表示一個URI。但它們的最后一個參數分別是Object[]和Map* 第三種的首個參數使用java.net.URI表示一個URI。且只有兩個參數

這是因為,String類型的URI支持占位符。比如:

restTemplate.getForObject("http://example.com/hotels/{hotel}/bookings/{booking}",String.class,"42", "21");

那么最終訪問的URI為:http://example.com/hotels/42/bookings/21

但是String有一個小缺陷:String形式的URI會被URL編碼兩次(URL encode請自行百度),這就要求服務器在獲取URI中的參數時主動進行一次解碼,但如果服務的提供者不這么做呢?

這時就需要使用不會使用任何編碼的java.net.URI

PS:參數‘Class<T>responseType'定義了返回數據的類型。

(3)Exchange與其它接口的不同:

>允許調用者指定HTTP請求的方法(GET,POST,PUT等)

>可以在請求中增加body以及頭信息,其內容通過參數‘HttpEntity<?>requestEntity'描述

>exchange支持‘含參數的類型'(即泛型類)作為返回類型,該特性通過‘ParameterizedTypeReference<T>responseType'描述。比如:

List<String> a = new ArrayList<String>(); 
System.out.println(a.getClass()); 
System.out.println(a.getClass().getGenericSuperclass()); 
ParameterizedTypeReference pt = new ParameterizedTypeReference<ArrayList<String>>() {}; 
System.out.println(pt.getType()); 

得到的結果是:

class java.util.ArrayList 
java.util.AbstractList<E> 
java.util.ArrayList<java.lang.String> 

* 這是因為ParameterizedTypeReference<ArrayList<String>>并不根據實參而是使用getGenericSuperclass()方法獲取其父類的類型(注意這里的new有花括號,是ParameterizedTypeReference的子類),父類的類型通過java.lang.reflect.Type描述,然后通過Type的getActualTypeArguments()獲得了父類的實參類型,注意得到的Type類,并不是class類。

(4)excute

所有的get、post、delete、put、options、head、exchange方法最終調用的都是excute方法。舉個栗子:

@Override 
public <T> T getForObject(String url, Class<T> responseType, Object urlVariables) throws RestClientException { 
 RequestCallback requestCallback =  acceptHeaderRequestCallback(responseType); 
 HttpMessageConverterExtractor<T> responseExtractor = new HttpMessageConverterExtractor<T>(responseType, getMessageConverters(), logger); 
 return execute(url, HttpMethodGET, requestCallback, responseExtractor, urlVariables); 
} 

Excute方法只是將String格式的URI轉成了java.net.URI,之后調用了doExecute方法。整個調用過程

Spring RestTemplate具體使用詳解

6.doExcute

6.1 定義

protected <T> T doExecute(URI url, HttpMethod method, RequestCallback requestCallback,ResponseExtractor<T> responseExtractor) throws RestClientException {…} 

這里需要了解兩個類: RequestCallback &ResponseExtractor

6.2RequestCallback

Callback interface for code that operates on a ClientHttpRequest. Allows to manipulate the request headers, and write to the request body.

簡單說:用于操作請求頭和body,在請求發出前執行。

該接口有兩個實現類:

AcceptHeaderRequestCallback 只處理請求頭,用于getXXX()方法。
HttpEntityRequestCallback 繼承于AcceptHeaderRequestCallback可以處理請求頭和body,用于putXXX()、postXXX()和exchange()方法。

* DELETE、HEAD、OPTIONS沒有使用這個接口。

6.3 發起請求

6.4ResponseExtractor

6.4.1 定義

Generic callback interface used by RestTemplate's retrieval methods Implementations of this interface perform the actual work of extracting data from a ClientHttpResponse, but don't need to worry about exception handling or closing resources.

簡單說:解析HTTP響應的數據,而且不需要擔心異常和資源的關閉。

該接口有三個實現類:

HeadersExtractor 用于提取請求頭。
HttpMessageConverterExtractor 用于提取響應body。
ResponseEntityResponseExtractor 使用HttpMessageConverterExtractor提取body(委托模式),然后將body和響應頭、狀態封裝成ResponseEntity對象。

6.4.2 提取響應body

提取分三步:

(1)提取器HttpMessageConverterExtractor尋找可用的轉化器在默認的RestTemplate的構造函數中初始化了轉化器集合,包括:

轉化器

可轉化的類型
ByteArrayHttpMessageConverter byte[]
StringHttpMessageConverter String
ResourceHttpMessageConverter Resource
SourceHttpMessageConverter javax.xml.transform.*
AllEncompassingFormHttpMessageConverter MultiValueMap
Jaxb2RootElementHttpMessageConverter XmlRootElement,XmlType(注解)
...  
MappingJackson2HttpMessageConverter Json

除了前五個,其他的轉化器會由classloader嘗試加載某個類來判斷工程是否包含某個包,而后決定是否加入轉化器集合。

提取器遍歷轉化器集合以查找可用的轉化器,其中MappingJackson2HttpMessageConverter總是在最后一個,因為該類實現了GenericHttpMessageConverter,算是一個通用轉化器,只有在找不到合適的轉化器時才輪到它。Spring提供了一個該類的實現,以保證總是能得到該類。

(2)轉化器尋找可用的反序列化器

轉化器持有一個反序列化器緩存集合,首先從緩存中尋找

如果已有可用的反序列化器,則直接返回。否則創建一個新的反序列化器。

反序列化器保存著待反序列化類的域、方法、構造器等信息,反序列化時就是使用構造器創建了一個新的實例。

以jackson為例,創建反序列化器的過程在jackson-databind-xxx.jar中,有興趣的可以看一下。調用棧如下(由下往上找):

BeanDeserializerFactory.addBeanProps/addObjectIdReader/addReferenceProperties/addInjectables
BeanDeserializerFactory.buildBeanDeserializer
BeanDeserializerFactory.createBeanDeserializer

(3)反序列化器執行反序列化

TOKEN

Json的一個或一組字符
START_OBJECT {
END_OBJECT }
START_ARRAY [
END_ARRAY ]
VALUE_TRUE true
VALUE_FALSE false
...  

Spring RestTemplate具體使用詳解

調用棧:

Spring RestTemplate具體使用詳解

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

广安市| 澄迈县| 德安县| 确山县| 大安市| 尼玛县| 萝北县| 乐安县| 永泰县| 靖安县| 双辽市| 闻喜县| 民丰县| 正阳县| 长海县| 富裕县| 稻城县| 西宁市| 漳浦县| 巩留县| 巫山县| 渑池县| 库车县| 庄浪县| 磴口县| 廉江市| 茌平县| 泰顺县| 天峨县| 湖口县| 安多县| 西乌珠穆沁旗| 成都市| 长宁区| 南阳市| 衡阳县| 普安县| 隆子县| 遵义市| 梧州市| 内江市|