Spring Cloud是目前用于开发微服务的主流框架之一,咱们都知道在微服务架构中最为基础、核心的模块,就是服务注册与发现。java
在Spring Cloud里咱们可使用它的Eureka模块来实现服务注册与发现,Spring Cloud Eureka是基于Netflix Eureka作了二次封装,它主要负责完成各个微服务实例的自动化注册和发现功能。web
Eureka由两个组件组成:spring
分布式系统中为何须要服务发现:apache
在实际的分布式环境下,架构规模每每再也不是几台服务器,而是每一个独立服务都跑在多台机器上。例如A服务部署在10台机器上,B服务也部署在10台机器上,C服务部署在5台机器上。服务器
如今A服务的一些功能须要调用B服务来实现,那么问题来了,A服务要如何才能调用B服务呢?一般状况下,咱们能够想到将B服务所在的全部机器地址,经过配置文件来配置到A服务中,使其可以经过配置好的地址去发现并调用B服务。架构
这也的确是一个可行的方法,可是这些机器的地址都是有可能发生变化的,并且在生产环境中也会出现部分服务宕机的状况,这样就有可能致使一些连锁效应。随着业务的扩展,机器也会愈来愈多,也没办法再手动经过配置文件这种方式来配置机器地址了。app
鉴于人类的懒惰天性。。。呸,鉴于人类对高效工做、美好生活的追求。因此专门用于服务注册与发现的工具被一一开发出来。有了服务治理框架后,服务发现就能够交由它来自动完成。这时候A服务只须要到注册中心进行服务注册,一样的B服务也到注册中心进行服务注册。注册以后,注册中心会经过相似心跳机制来确认服务的存活。若是确认某个服务宕机后,注册中心会把宕机的服务剔除掉。当A服务要调用B服务的时候,则到注册中心去获取B服务的调用地址便可,B服务调用A服务也是同理。注册中心就至关于一个服务与服务之间的桥梁或者说中间人,能够说帮咱们管理了服务之间杂事。框架
Eureka服务治理体系以下:maven
Spring Cloud官网地址以下:分布式
废话很少说,本小节咱们来搭建一个Eureka Server,即服务注册中心。打开IDEA,新建一个Spring Initializr项目:
勾选Eureka Server模块:
完成项目的建立:
项目生成的pom.xml文件内容以下:
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>org.zero</groupId> <artifactId>eureka</artifactId> <version>0.0.1-SNAPSHOT</version> <packaging>jar</packaging> <name>eureka</name> <description>Demo project for Spring Boot</description> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.0.4.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> <java.version>1.8</java.version> <spring-cloud.version>Finchley.SR1</spring-cloud.version> </properties> <dependencies> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> </dependencies> <dependencyManagement> <dependencies> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-dependencies</artifactId> <version>${spring-cloud.version}</version> <type>pom</type> <scope>import</scope> </dependency> </dependencies> </dependencyManagement> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build> </project>
注:SpringBoot与SpringCloud组件的版本是有一个对应关系的,这个在官网上有详细的对照图。值得注意的是,SpringCloud不是以2.0.1这种数字的方式来标识版本号,而是以伦敦地铁站的名称来标识版本号,而且这些名称是符合字母顺序的。
项目建立好后,咱们能够试着启动看看,可是在启动以前须要在启动类中加上@EnableEurekaServer
注解,表示启用Eureka Server,不然访问就会报404。代码以下:
package org.zero.eureka; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer; @SpringBootApplication @EnableEurekaServer public class EurekaApplication { public static void main(String[] args) { SpringApplication.run(EurekaApplication.class, args); } }
启动项目后,可以访问到以下Eureka注册中心的页面表明成功:
此时虽然可以正常访问到注册中心的页面,可是会发现控制台一直在报错,提示Cannot execute request on any known server
。这是由于这个Eureka Server既是server的同时,也是一个client,它也是须要把本身注册到一个注册中心去的。由于咱们并无配置注册中心的地址,因此它没办法注册本身就会报这个错误。
既然如此,咱们只需配置一下注册中心的地址便可,编辑application.yml配置文件内容以下:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/ # 指定注册中心的url register-with-eureka: false # 指定不进行注册操做,默认为true,若进行注册的话,会显示在Eureka信息面板上 server: enable-self-preservation: false # 禁用eureka server的自我保护机制,建议在生产环境下打开此配置 spring: application: name: eureka # 指定应用的名称 server: port: 8761 # 指定项目的端口号
注:因为server和client是采用心跳机制来确认存活的,因此在启动项目的过程可能依旧会报错。可是只要启动后不是一直报错,而且能正常访问Eureka信息面板页面的话,则表明项目是在正常运行的
在上一小节中,咱们简单介绍了如何建立、配置Eureka Server项目。既然咱们已经知道了如何搭建Eureka Server,那么本小节咱们将介绍Eureka Client的使用,会简单演示一下如何经过Eureka Client进行服务注册。
一样的,使用IDEA建立一个Spring Initializr项目,只不过在勾选模块的时候须要选择Eureka Discovery,以下:
项目生成的pom.xml文件内容以下:
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>org.zero.eureka</groupId> <artifactId>client</artifactId> <version>0.0.1-SNAPSHOT</version> <packaging>jar</packaging> <name>client</name> <description>Demo project for Spring Boot</description> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.0.4.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> <java.version>1.8</java.version> <spring-cloud.version>Finchley.SR1</spring-cloud.version> </properties> <dependencies> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> </dependencies> <dependencyManagement> <dependencies> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-dependencies</artifactId> <version>${spring-cloud.version}</version> <type>pom</type> <scope>import</scope> </dependency> </dependencies> </dependencyManagement> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build> </project>
项目的依赖都加载完成后,在启动类中加上@EnableDiscoveryClient
,声明这是一个eureka client,不然不会进行服务注册:
package org.zero.eureka.client; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cloud.client.discovery.EnableDiscoveryClient; @SpringBootApplication @EnableDiscoveryClient public class ClientApplication { public static void main(String[] args) { SpringApplication.run(ClientApplication.class, args); } }
接着就是在application.yml配置文件中,配置注册中心即eureka server的地址,以及项目的名称和启动端口号。以下:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/ spring: application: name: eureka-client server: port: 9088
完成以上配置后,便可启动项目。可是我这里启动项目的时候失败了,控制台输出以下警告信息:
Invocation of destroy method failed on bean with name 'scopedTarget.eurekaClient': org.springframework.beans.factory.BeanCreationNotAllowedException: Error creating bean with name 'eurekaInstanceConfigBean': Singleton bean creation not allowed while singletons of this factory are in destruction (Do not request a bean from a BeanFactory in a destroy method implementation!)
这是由于client里不包含Tomcat的依赖,因此Spring容器没法建立一些实例,从而致使项目没法启动,只需在pom.xml文件中,加上web依赖便可:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency>
项目启动成功后,能够在eureka server的信息面板中查看到已注册的实例信息,以下:
高可用是在服务架构设计中,频繁出现的词汇。微服务架构里天然也同样须要保证服务的高可用性,因此本小节将简单说明一下Eureka是如何实现高可用的。
在实际生产环境中服务器是很脆弱的,单台服务器确定是没法知足高可用的需求,为了保证高可用性咱们一般会准备多台服务器。但能够发现上文中所搭建的eureka server是单机的,若这个eureka server宕机,则会致使与之关联的所有微服务发生故障。
既然单机没法保证高可用,那么咱们就加多一台机器好了,而后让这两个eureka server互相进行关联。例如我如今有两台eureka server。一台名叫eureka-server01跑在8761端口上,另外一台名叫eureka-server02跑在8762端口上。而后只须要两个步骤便可实现高可用:
1.编辑两台eureka server的配置文件,eureka-server01:
eureka: client: service-url: defaultZone: http://localhost:8762/eureka/ # 指向eureka-server02的url register-with-eureka: false server: enable-self-preservation: false spring: application: name: eureka-server01 server: port: 8761
eureka-server02:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/ # 指向eureka-server01的url register-with-eureka: false server: enable-self-preservation: false spring: application: name: eureka-server02 server: port: 8762
2.编辑eureka client的配置文件,多个url使用英文逗号分隔:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/,http://localhost:8762/eureka/ spring: application: name: eureka-client server: port: 9088
若是项目规模比较大,有两个以上的eureka server,那该如何在配置文件中配置呢?其实只须要每台eureka server分别配置除本身以外的eureka server机器,而后eureka client则配置全部的eureka server地址便可。以下图:
配置文件示例,eureka-server01:
eureka: client: service-url: defaultZone: http://localhost:8762/eureka/,http://localhost:8763/eureka/ register-with-eureka: false spring: application: name: eureka-server01 server: port: 8761
配置文件示例,eureka-server02:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/,http://localhost:8763/eureka/ register-with-eureka: false spring: application: name: eureka-server02 server: port: 8762
配置文件示例,eureka-server03:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/,http://localhost:8762/eureka/ register-with-eureka: false spring: application: name: eureka-server03 server: port: 8763
eureka client的配置文件示例:
eureka: client: service-url: defaultZone: http://localhost:8761/eureka/,http://localhost:8762/eureka/,http://localhost:8763/eureka/ spring: application: name: eureka-client server: port: 9088