在前文中我們已經搭建了高可用的註冊中心,並向註冊中心註冊了兩個服務,hello-service,現在已經有了服務的提供方,那麼自然也要有服務的消費方,這篇文章就來搭建一個服務消費者,可以發現並且消費服務。服務的發現是由Eureka的客戶端完成的,而服務的消費是由Ribbon完成的。Ribbon是一個負載均衡器,在客戶端配置了ribbonServerList後即可達到負載均衡的效果,先不深究Ribbon的具體實現原理,我們只需要理解它可以做到負載均衡,配合Eureka的服務發現機制可以達到選擇服務實例的目的從而實現服務消費
首先我們啓動前文中搭建的兩個註冊中心和兩個hello-service實例,啓動後在註冊中心的管理界面應該可以看到如下的服務:
接下來仿照前面的項目,再創建一個新項目叫做ribbon-consumer,配置同樣非常簡單,在啓動類上加入註解@EnableDiscoveryClient即可讓應用註冊爲Eureka的客戶端並具有發現服務的能力,同時在啓動類中注入RestTemplate,並通過註解@LoadBalance註解開啓該客戶端的負載均衡功能,啓動類代碼如下:
@EnableDiscoveryClient
@SpringBootApplication
public class ConsumerApplication {
@Bean
@LoadBalanced
RestTemplate restTemplate() {
return new RestTemplate();
}
public static void main(String[] args) {
SpringApplication.run(ConsumerApplication.class, args);
}
}
配置了啓動類之後我們再新建一個ConsumerController類,在該類中實現一個/ribbon-consumer接口,該接口通過RestTemplate調用hello-service的/hello接口,注意在代碼中我們只需要指定service name,即HELLO-SERVICE,而並不需要輸入具體的host和port,這也是服務治理框架中一個非常重要的特性,你需要知道的是服務,而不是背後具體的服務器,否則的話就和一個http調用的rpc接口請求沒什麼區別了,controller參考代碼如下:
@RestController
public class ConsumerController {
@Autowired
RestTemplate restTemplate;
@RequestMapping(value = "/ribbon-consumer")
public String helloConsumer() {
return restTemplate.getForEntity("http://hello-service/hello", String.class).getBody();
}
}
當然最後不要忘記在配置文件application.yml中加入Eureka註冊中心客戶端的相關配置,注意端口號不要和前面的其他項目衝突:
spring:
application:
name: ribbon-consumer
server:
port: 3333
eureka:
client:
service-url:
defaultZone: http://localhost:1111/eureka
全部配置完成後啓動ribbon-consumer項目,在Eureka的信息面板頁面應該會發現多了一個ribbon-consumer應用,至此說明我們的服務發現和消費服務已經成功註冊到了註冊中心,接下來我們需要測試一下,不過在測試前我們可以優化一下之前hello-service的返回值,由於ribbon-consumer採用了負載均衡,而且我們有兩個hello-service提供服務,所以我們可以通過在接口返回中打印具體的server端口號來區分調用被髮送到了哪個server上,所以這裏我們稍微改一下之前的/hello接口,在項目log和返回值中都加入一些信息:
@RestController
public class HelloController {
private final Logger logger = LoggerFactory.getLogger(getClass());
@Autowired
private DiscoveryClient client;
@Autowired
private Registration registration;
@Autowired
private ServerConfig serverConfig;
@RequestMapping(value = "/hello", method = RequestMethod.GET)
public String index() {
List<ServiceInstance> instances = client.getInstances(registration.getServiceId());
if (CollectionUtils.isEmpty(instances)) {
return null;
}
int port = serverConfig.getServerPort();
for (ServiceInstance si : instances) {
if (si.getPort() == port) {
logger.info("/hello, host :" + si.getHost() + ", service_id :" + si.getServiceId());
return "Hello World @" + port;
}
}
return null;
}
}
現在我們重啓一下hello-service,然後可以直接在瀏覽器中訪問http://localhost:3333/ribbon-consumer ,我們可以看到返回了Hello World @2221, 我們重新刷新一下頁面,會發現變成了Hello World @2222,說明我們不僅成功的發現服務並且進行了
另外有興趣的朋友可以觀察一下各個項目控制檯中的日誌,爲我們打印了關於請求數,連接數,連接信息,失敗信息等有用的信息供大家在實際工作學習排查問題。那麼至此,微服務框架就已經具有了一個高可用註冊中心,兩個服務提供者hello-service,和一個服務發現者ribbon-consumer。