RestTemplate
是 Spring Resources 中一個訪問第三方 RESTful API 接口的網絡請求框架。RestTemplate 的設計原則和其他 Spring Template(例如 JdbcTemplate、JmsTemplate)類似,都是為執行復雜任務提供了一個具有默認行為的簡單方法。
RestTemplate 是用來消費 REST 服務的,所以 RestTemplate 的主要方法都與 REST 的 Http 協議的一些方法緊密相連,例如 HEAD、GET、POST、PUT、DELETE 和 OPTIONS 等方法,這些方法在 RestTemplate 類對應的方法為 headForHeaders()、getForObject()、postForObject()、put() 和 delete() 等。
舉例說明,寫一個 RestTestController 類,獲取 https://www.baidu.com/ 的網頁 Html 代碼。首先在 RestTestController 類上加 @RestController 注解,開啟 RestController 的功能。通過 RestTemplate 的 getForObject() 方法可以獲取 https://www.baidu.com/ 的網頁 Html 代碼,并在 API 接口 /testRest 返回該網頁的 Html 字符串。代碼如下:
@RestController
public class RestTestController {
@GetMapping("/testRest")
public String testRest(){
RestTemplate restTemplate = new RestTemplate();
return restTemplate.getForObject("https://www.baidu.com/", String.class);
}
}
RestTemplate 支持常見的 Http 協議的請求方法,例如 Post、Put、Delete 等,所以用 RestTemplate 很容易構建 RESTful API。在上面的例子中,RestTemplate 用 Get 方法獲取 https://www.baidu.com 網頁的 Html 字符串。RestTemplate 的使用很簡單,它支持 Xml、JSON 數據格式,默認實現了序列化,可以自動將 JOSN 字符串轉換為實體。例如以下代碼可以將返回的 JSON 字符串轉換成一個 User 對象。
User user=restTemplate.getForObject("https://www.xxx.com/",User.class);
Ribbon 簡介
負載均衡是指將負載分攤到多個執行單元上,常見的負載均衡有兩種方式。服務端負載均衡, 是獨立進程單元,通過負載均衡策略,將請求轉發到不同的執行單元上,例如 Ngnix。另一種是客戶端負載均衡, 是將負載均衡邏輯以代碼的形式封裝到服務消費者的客戶端上,服務消費者客戶端維護了一份服務提供者的信息列表,有了信息列表,通過負載均衡策略將請求分攤給多個服務提供者,從而達到負載均衡的目的。
Ribbon 是 Netflix 公司開源的一個負載均衡的組件,它屬于上述的客戶端負載均衡,是將負載均衡邏輯封裝在客戶端中,并且運行在客戶端的進程里。Ribbon 是一個經過了云端測試的 IPC 庫,可以很好地控制 HTTP 和 TCP 客戶端的負載均衡行為。
在 Spring Cloud 構建的微服務系統中,Ribbon 作為服務消費者的負載均衡器,有兩種使用方式,一種是和 RestTemplate 相結合,另一種是和 Feign 相結合。Feign 已經默認集成了 Ribbon,關于 Feign 的內容將會在下一章進行詳細講解。
Ribbon 有很多子模塊,但很多模塊沒有用于生產環境,目前 Netflix 公司用于生產環境的 Ribbon 子模塊如下:
- ribbon-loadbalancer:可以獨立使用或與其他模塊一起使用的負載均衡器 API。
- ribbon-eureka:Ribbon 結合 Eureka 客戶端的 API,為負載均衡器提供動態服務注冊列表信息。
- ribbon-core:Ribbon 的核心 API。
本節的實驗會基于上一節 Eureka Server 實驗代碼的基礎上進行改造。首先回顧一下上一節實驗中的代碼結構,它包括一個服務注冊中心 eureka-server、一個服務提供者 eureka-client。eureka-client 向 eureka-server 注冊服務,并且 eureka-client 提供了一個 /hi API 接口,用于提供服務。
首先修改 eureka-server 的 application.yml 文件,修改后的具體配置代碼如下:
server:
port: 8080
eureka:
instance:
hostname: localhost
client:
registerWithEureka: false
fetchRegistry: false
serviceUrl:
defaultZone: http://${eureka.instance.hostname}:${server.port}/eureka/
然后修改 eureka-client 的 application.yml 文件,修改后的具體配置代碼如下:
---
spring:
profiles: client01
application:
name: eureka-client
server:
port: 8762
eureka:
client:
service-url:
defaultZone: http://localhost:8080/eureka/
---
spring:
profiles: client02
application:
name: eureka-client
server:
port: 8763
eureka:
client:
service-url:
defaultZone: http://localhost:8080/eureka/
上述配置中,定義了兩個 profile 文件,分別為 client01 和 client02,它們的端口號分別為 8762 和 8763,之后我們可以根據不同的 profile 來啟動兩個 eureka-client。
修改完 eureka-client 的配置文件之后,進入 eureka-client 根目錄,通過 Maven 編譯打包工程。
cd /home/project/eureka/eureka-client
mvn clean package -Dmaven.test.skip=true
通過 java -jar 的方式啟動工程,并通過 spring.profiles.active 指定啟動的配置文件,在本實驗中需要啟動兩個 Eureka Client 實例。由于實驗可能會啟動很多個 Spring Boot 工程,為了防止內存溢出造成預料之外的后果,從本節實驗開始,都通過使用 java -jar 的形式啟動 Spring Boot 工程,這也是真實環境中所采用的方法。同時避免了開啟多個控制臺窗口。
首先在控制臺中進入主 Maven 工程 eureka 的根目錄,通過以下命令打包 eureka-server 和 eureka-client 模塊。
cd /home/project/eureka
mvn clean package -Dmaven.test.skip=true
java -Xmx128m -jar ./eureka-server-0.0.1-SNAPSHOT.jar > server.log 2>&1 &
tail -n 20 -f server.log
Eureka Server 啟動完成后,通過以下命令后臺啟動 Eureka Client。
java -Xmx128m -jar ./eureka-client-0.0.1-SNAPSHOT.jar --spring.profiles.active=client01 > client01.log 2>&1 &
java -Xmx128m -jar ./eureka-client-0.0.1-SNAPSHOT.jar --spring.profiles.active=client02 > client02.log 2>&1 &
Eureka Server 和兩個 Eureka Client 都啟動完成后,打開 Web 服務,可以在 Eureka Server 的首頁看到兩個 eureka-client 實例已經成功向服務注冊中心注冊,它們的端口分別為 8762 和 8763,如下圖所示: