什么是Spring Cloud Eureka
Eureka是Netflix公司開發的開源服務注冊發現組件,服務發現可以說是微服務開發的核心功能了,微服務部署后一定要有服務注冊和發現的能力,Eureka就是擔任這個角色。如果你用過Dubbo的話,Dubbo里服務注冊和發現就是通過Zookeeper框架完成的。
Eureka 目前是2.2.x版本,目前官方已經宣布不再維護和更新了,不過Eureka 做注冊中心已經在生產環境中大規模使用了,可以說很穩定了。從我個人的角度看,目前大家使用的更多的是阿里的Nacos和Consul 這兩個組件實現了不止服務發現和注冊,微服務開發不用再去依賴更多的組件和框架。這篇文章模擬一下Eureka Server集群和服務提供者集群和服務消費。
版本說明
SpringCloud + SpringBoot開發微服務並不是版本越新越好,Spring Cloud官方提供了一個版本對應關系。目前最新的就是Hoxton, 對應SpringBoot 2.2.x版本。
准備工作
- 新建父工程, 主要約定SpringCloud, SpringBoot版本號,我使用的是Hoxton.SR1, SpringBoot2.2。
- 新建5個子Module,這里兩個Eureka Server,兩個Service,一個consumer, 兩個Eureka Servr我用7001和7002端口來模擬和區分Eureka集群,兩個Service 用8001和8002來模擬,主要是想在服務消費時反應客戶端負載均衡。
項目結構如下圖
父工程pom.xml
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.2.2.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>Hoxton.SR1</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
服務注冊中心
服務提供者我用了7001和7002模擬集群,兩個程序代碼都是相同。
新建Module,添加spring-cloud-starter-netflix-eureka-server。
pom.xml
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
</dependencies>
application.yml
server:
port: 7001
eureka:
client:
service-url:
defaultZone: http://eureka7002.com:7002/eureka # 集群就是指向其他配置中心
register-with-eureka: false
fetch-registry: false
instance:
hostname: eureka7001.com #eureka服務端實例名稱
server:
enable-self-preservation: false
修改啟動類
@SpringBootApplication
@EnableEurekaServer
public class MyEurekaServer7001 {
public static void main(String[] args) {
SpringApplication.run(MyEurekaServer7001.class,args);
}
}
我這里貼出了7001的代碼,7002唯一不同的地方在application.yml配置里,如果是集群的話eureka.client.service-url.defaultZone需要配置集群里其他server節點。這里instance.hostname配置為eureka7001.com, 對應配置需要修改下本機hosts
127.0.0.1 eureka7001.com
127.0.0.1 eureka7002.com
啟動兩個注冊中心服務
eureka7001
eureka7002
服務啟動后,注意到DS Replicas中包含集群中另一台機器,說明集群已生效。
服務提供者
服務提供者我這里用了8001和8002兩個來模擬集群。
pom.xml
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
</dependencies>
application.yml
server:
port: 8001
spring:
application:
name: CLOUD-STUDENT-SERVICE
eureka:
client:
register-with-eureka: true
#集群節點下需要設置為true,才能配合客戶端使用Ribbon 負載均衡
fetch-registry: true
service-url:
defaultZone: http://eureka7001.com:7001/eureka, http://eureka7002.com:7002/eureka #集群需要配置所有注冊中心
instance:
instance-id: student-service8001
prefer-ip-address: true # 訪問路徑可以顯示ip地址
添加一個服務接口
@RestController
@RequestMapping("/student")
public class StudentController {
@GetMapping("/list")
public List<String> list(){
return Arrays.asList("Kobe","Lin","Tim","James");
}
@GetMapping("/version")
public String version(){
return "8001,202007162310";
}
}
修改啟動類
@SpringBootApplication
@EnableEurekaClient
public class MyStudentService8001 {
public static void main(String[] args) {
SpringApplication.run(MyStudentService8001.class,args);
}
}
依次啟動兩個服務提供者8001和8002。服務啟動后無錯誤的情況下刷新Eureka Server界面。
eureka7001
eureka7002
在Application里已經看到CLOUD-STUDENT-SERVICE的狀態中已經有兩個服務注冊進來了。
服務消費者
服務消費者我就寫了一個Module,端口8087
pom.xml
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
</dependencies>
application.yml
這里客戶端消費服務時需要配置集群中的兩個注冊中心地址。
server:
port: 8087
spring:
application:
name: student-consumer
eureka:
client:
fetch-registry: true
register-with-eureka: false
service-url:
defaultZone: http://eureka7001.com:7001/eureka, http://eureka7002.com:7002/eureka
服務消費接口
服務消費我們仍然用的RestTemplate,客戶端負載均衡用的實際上是Ribbon組件,默認使用輪訓算法。
@RestController
@RequestMapping("/student")
public class StudentController {
@Autowired
RestTemplate restTemplate;
@GetMapping("/version")
public String index(){
return restTemplate.getForObject("http://CLOUD-STUDENT-SERVICE/student/version",String.class);
}
}
修改啟動類
我在啟動類還配置了RestTemplate,啟動類兩個關鍵注解SpringBootApplication和EnableEurekaClient。
@SpringBootApplication
@EnableEurekaClient
public class StudentConsumer8087 {
@Bean
@LoadBalanced
public RestTemplate restTemplate(){
return new RestTemplate();
}
public static void main(String[] args) {
SpringApplication.run(StudentConsumer8087.class,args);
}
}
啟動並訪問服務消費者接口http://localhost:8087/student/version, 從輸出結果能看到會輪訓調用8001和8002的接口8002,202007162310,8001,202007162310。