摘要:調(diào)用的默認構(gòu)造函數(shù),對象在底層通過使用包下的實現(xiàn)創(chuàng)建請求,可以通過使用指定不同的請求方式。接口主要提供了兩種實現(xiàn)方式一種是,使用提供的方式既包提供的方式創(chuàng)建底層的請求連接。
自從RESTFul API興起后,Spring就給開發(fā)者提供了一個訪問Rest的客服端,RestTemplate不僅可以很方便的調(diào)用http接口,還可以調(diào)用同一注冊中心下的微服務(wù),同時還有負載均衡和熔斷機制。當然我也聽說OKhttp,HTTPClient之類的網(wǎng)絡(luò)框架,但是我們這里重點講的是RestTemplate(其中一種實現(xiàn)就是基于HTTPClient實現(xiàn)的),因為這是SpringBoot自帶,而且基本使用場景都覆蓋了。什么是RestTemplate?
RestTemplate是Spring提供的用于訪問Rest服務(wù)的客戶端,
RestTemplate提供了多種便捷訪問遠程Http服務(wù)的方法,能夠大大提高客戶端的編寫效率。
調(diào)用RestTemplate的默認構(gòu)造函數(shù),RestTemplate對象在底層通過使用java.net包下的實現(xiàn)創(chuàng)建HTTP 請求,
可以通過使用ClientHttpRequestFactory指定不同的HTTP請求方式。
ClientHttpRequestFactory接口主要提供了兩種實現(xiàn)方式
1、一種是SimpleClientHttpRequestFactory,使用J2SE提供的方式(既java.net包提供的方式)創(chuàng)建底層的Http請求連接。
2、一種方式是使用HttpComponentsClientHttpRequestFactory方式,底層使用HttpClient訪問遠程的Http服務(wù),使用HttpClient可以配置連接池和證書等信息。
簡單配置RestTemplate@Configuration class RestTemplateConfig { /** * 構(gòu)建一個使用默認配置RestTemplate Bean */ @Bean fun restTemplate():RestTemplate { return RestTemplateBuilder().build() } }發(fā)送POST請求,通過Form表單提交參數(shù)
@Component class RestTemplateClient { @Autowired lateinit var restTemplate: RestTemplate val logger:Logger = LoggerFactory.getLogger(this.javaClass) /** * 發(fā)送POST請求,通過Form表單提交參數(shù) */ fun編寫Controller及測試postForm(params: Map , url: String, t:T) : ResultDTO ? { logger.info("請求URL===========:{}", url) logger.info("請求參數(shù)===========:{}", params) val httpHeaders = HttpHeaders() httpHeaders.contentType = MediaType.APPLICATION_FORM_URLENCODED val requestParams = LinkedMultiValueMap () params.forEach(requestParams::add) val httpEntity = HttpEntity >(requestParams, httpHeaders) return restTemplate.postForObject(url, httpEntity, ResultDTO ().javaClass) } }
@RestController @RequestMapping("/api") class LotteryController { @Autowired lateinit var restTemplateClient: RestTemplateClient /** * 彩票類型查詢 */ @PostMapping("/lottery/types") fun lotteryTypes(@RequestBody params:Map發(fā)送POST請求,通JSON傳參) : ResultDTO ? { return restTemplateClient.postForm(params, UrlEnums.DEMO1.url, LotteryDTO()) } }
/** * 發(fā)送POST請求,通過Form表單提交參數(shù) */ fun發(fā)送GET請求postBody(params: Map , url: String, t:T) : ResultDTO ? { logger.info("請求URL===========:{}", url) logger.info("請求參數(shù)===========:{}", params) val httpHeaders = HttpHeaders() httpHeaders.contentType = MediaType.APPLICATION_JSON_UTF8 val httpEntity = HttpEntity(params, httpHeaders) return restTemplate.postForObject(url, httpEntity, ResultDTO ().javaClass) }
/** * 發(fā)送POST請求,通過Form表單提交參數(shù) */ funGET下載資源get(params: Map , url: String, t:T) : ResultDTO ? { logger.info("請求URL===========:{}", url) logger.info("請求參數(shù)===========:{}", params) val builder = StringBuilder() params.forEach{k, v -> builder.append(k).append("=").append(v).append("&") } val param : String = builder.toString().substring(0, builder.length - 1) return restTemplate.getForObject(url.plus(param), ResultDTO ().javaClass) }
/** * 下載資源 */ fun downGet(url: String, headerParams: Map調(diào)用微服務(wù)接口) : ByteArray? { logger.info("資源URL==========>:{}", url) val httpHeaders = HttpHeaders() if (!CollectionUtils.isEmpty(headerParams)) { headerParams.forEach(httpHeaders::add) } val httpEntity = HttpEntity(LinkedMultiValueMap ().putAll(httpHeaders)) return restTemplate.exchange(url, HttpMethod.GET, httpEntity, ByteArray::class.java).body }
我們在開發(fā)微服務(wù)應(yīng)用的時候,有時候一個應(yīng)用可能會部署很多臺,而且還是不在不通的機器上,這個時候我們可以用Nginx做一個負載均衡,但是我們這里推薦使用的是SpringCloud提供的負載均衡,只要在Bean上面加一個注解即可@LoadBalanced,如下所示:
@Bean @LoadBalanced fun restTemplate():RestTemplate { return RestTemplateBuilder().build() }
和調(diào)用普通的http請求有點不同的地方是,調(diào)用微服務(wù)是http://application.name/xxx,我們訪問的路徑是微服務(wù)應(yīng)用名稱+業(yè)務(wù)路由,因為服務(wù)會注冊到注冊中心,注冊中心會維護這個應(yīng)用名稱,當我們通過應(yīng)用去訪問時,就可以知道這個應(yīng)用在那臺機器上了。
基礎(chǔ)使用講完了,但是這里還存在一個問題,因為我們使用的是最簡單的配置,在這里我們沒有設(shè)置默認超時時間,官方默認的是60S,但是我們業(yè)務(wù)中通常對響應(yīng)速度是有要求,一分鐘的等待時間對于用戶來說太長了,這個時候我們就需要有一個超時機制,具體的設(shè)置如下:
package io.intodream.kotlin05.config import org.apache.http.impl.client.HttpClientBuilder import org.springframework.beans.factory.annotation.Value import org.springframework.boot.context.properties.ConfigurationProperties import org.springframework.boot.web.client.RestTemplateBuilder import org.springframework.context.annotation.Bean import org.springframework.context.annotation.Configuration import org.springframework.http.client.ClientHttpRequestFactory import org.springframework.http.client.HttpComponentsClientHttpRequestFactory import org.springframework.http.client.SimpleClientHttpRequestFactory import org.springframework.http.converter.HttpMessageConverter import org.springframework.http.converter.StringHttpMessageConverter import org.springframework.stereotype.Component import org.springframework.web.client.RestTemplate import java.nio.charset.StandardCharsets /** * @description * RestTemplate 配置 * @author Jwenk * @copyright intoDream.io 筑夢科技 * @email [email protected] * @date 2019-04-05,22:42 */ @Configuration class RestTemplateConfig { @Value("${remote.maxTotalConnect}") val maxTotalConnect: Int = 0 @Value("${remote.maxConnectPerRoute}") val maxConnectPerRoute: Int = 200 @Value("${remote.connectTimeout}") val connectTimeout: Int = 2000 @Value("${remote.readTimeout}") val readTimeout: Int = 3000 /** * 創(chuàng)建HTTP客戶端工廠 */ private fun createFactory() : ClientHttpRequestFactory { if (this.maxTotalConnect <= 0) { val factory = SimpleClientHttpRequestFactory() factory.setReadTimeout(this.readTimeout) factory.setConnectTimeout(this.connectTimeout) return factory } val httpClient = HttpClientBuilder.create() .setMaxConnTotal(this.maxTotalConnect) .setMaxConnPerRoute(this.maxConnectPerRoute).build() val factory = HttpComponentsClientHttpRequestFactory(httpClient) factory.setConnectTimeout(this.connectTimeout) factory.setReadTimeout(this.readTimeout) return factory } /** * 構(gòu)建一個使用默認配置RestTemplate Bean */ @Bean fun restTemplate():RestTemplate { val restTemplate = RestTemplate(this.createFactory()) val converterList : MutableList> = restTemplate.messageConverters var converterTarget : HttpMessageConverter<*>? = null /** * 重新設(shè)置StringHttpMessageConverter字符集為UTF-8,解決中文亂碼問題 */ for (item: HttpMessageConverter<*> in converterList) { if (StringHttpMessageConverter::class.java == item.javaClass) { converterTarget = item break } } if (null != converterTarget) { converterList.remove(converterTarget) } converterList.add(1, StringHttpMessageConverter(StandardCharsets.UTF_8)) return restTemplate } }
記得在pom.xml里面引入HttpClient的jar,不然啟動會報錯的
org.apache.httpcomponents httpclient 4.5.6
關(guān)于RestTemplate的一些使用及配置優(yōu)化到此就結(jié)束了,喜歡的朋友可以關(guān)注本人的博客https://www.tisnz.com,如果有不對的地方歡迎指正。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/74113.html
摘要:一本節(jié)目標前兩章主要講了的基本操作,這一章我們將學(xué)習(xí)使用訪問,并通過完成簡單操作。這里有一個問題什么不選用數(shù)據(jù)庫呢答案是目前支持。突出點是,即非阻塞的。二構(gòu)建項目及配置本章不在講解如何構(gòu)建項目了,大家可以參考第一章。 showImg(https://segmentfault.com/img/remote/1460000018819338?w=1024&h=500); 一、本節(jié)目標 前兩...
摘要:一本節(jié)目標前兩章主要講了的基本操作,這一章我們將學(xué)習(xí)使用訪問,并通過完成簡單操作。這里有一個問題什么不選用數(shù)據(jù)庫呢答案是目前支持。突出點是,即非阻塞的。二構(gòu)建項目及配置本章不在講解如何構(gòu)建項目了,大家可以參考第一章。 showImg(https://segmentfault.com/img/remote/1460000018819338?w=1024&h=500); 一、本節(jié)目標 前兩...
摘要:在很多服務(wù)中我經(jīng)常需要用到發(fā)送郵件功能,所幸的是可以快速使用的框架,只要引入改框架我們可以快速的完成發(fā)送郵件功能。引入獲取郵件發(fā)送服務(wù)器配置在國內(nèi)用的最多的就是郵件和網(wǎng)易郵件,這里會簡單講解獲取兩家服務(wù)商的發(fā)送郵件配置。 showImg(https://segmentfault.com/img/remote/1460000018819338?w=1024&h=500); 在很多服務(wù)中我...
摘要:二教程環(huán)境三創(chuàng)建項目創(chuàng)建項目有兩種方式一種是在官網(wǎng)上創(chuàng)建二是在上創(chuàng)建如圖所示勾選然后點,然后一直默認最后點擊完成即可。我們這里看到和普通的接口沒有異同,除了返回類型是用包裝之外。與之對應(yīng)的還有,這個后面我們會講到。 showImg(https://segmentfault.com/img/remote/1460000018819338?w=1024&h=500); 從去年開始就開始學(xué)習(xí)...
摘要:一個客戶端請求從發(fā)出到被響應(yīng)經(jīng)歷了哪些組件哪些微服務(wù)請求總時長每個組件所花時長等信息我們有必要了解和收集,以幫助我們定位性能瓶頸進行性能調(diào)優(yōu),因此監(jiān)控整個微服務(wù)架構(gòu)的調(diào)用鏈十分有必要,本文將闡述如何使用搭建微服務(wù)調(diào)用鏈追蹤中心。 showImg(https://segmentfault.com/img/remote/1460000014553707); 概述 一個完整的微服務(wù)系統(tǒng)包含...
閱讀 3984·2021-11-24 10:46
閱讀 1842·2021-11-16 11:44
閱讀 2330·2021-09-22 16:02
閱讀 1445·2019-08-30 15:55
閱讀 1157·2019-08-30 12:46
閱讀 594·2019-08-28 18:31
閱讀 2822·2019-08-26 18:38
閱讀 1122·2019-08-23 16:51