前言
这周一入职了新公司,主要负责Java后端开发工作。目前正在开展一个全新的业务,技术选型为SpringCloud全家桶,项目的骨架由我负责搭建。由于前几家公司的微服务框架都是使用Dubbo + SpringBoot,然后平时对SpringCloud 这一套了解不多。这两天正好有时间可以研究下SpringCloud如何使用、SpringCloud 如何与其他组件整合,后面计划弄一个SpringCloud整合系列出来。
一、SpringCloud简介
SpringCloud 是基于SpringBoot提供一整套的微服务解决方案,包括服务注册与发现、配置中心、消息总线、服务网关、负载均衡、熔断器、全链路监控等组件。除了Netfix提供的开源组件外,还有一些第三方的开源组件可以使用。
开发者可以很方便的将微服务中常用组件无缝整合进SpringCloud中,然后通过SpringBoot风格进行封装和屏蔽掉底层复杂的配置和实现原理,非常容易上手。简单来说,SpringCloud 就是分布式微服务的一站式解决方案,俗称SpringCloud全家桶。
二、SpringCloud和SpringBoot的关系
- SpringBoot: 专注于方便快速开发单个微服务系统。
- SpringCloud: 微服务系统协调治理框架,就是将SpringBoot开发的多个微服务管理起来。
- SpringBoot 可以不依赖SpringCloud独立使用,而SpringCloud的使用一定离不开SpringBoot。
三、项目搭建
3.1 环境准备
- 开发工具IDEA
- JDK1.8
- Apache Maven 3.5.2,使用阿里云镜像
- SpringBoot 2.1.3.RELEASE 版本
- SpringCloud Greenwich.SR1 版本
3.2 父项目
创建一个名称为 spring-cloud-parent 的pom项目。用来管理SpringBoot、SpringCloud,及其他各种jar 包依赖。pom文件内容如下:
<?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>com.gallenzhang</groupId>
<artifactId>spring-cloud-parent</artifactId>
<version>1.0.0-SNAPSHOT</version>
<packaging>pom</packaging>
<name>spring-cloud-parent</name>
<properties>
<spring-boot.version>2.1.3.RELEASE</spring-boot.version>
<spring-cloud.version>Greenwich.SR1</spring-cloud.version>
<spring-cloud-alibaba.version>2.1.0.RELEASE</spring-cloud-alibaba.version>
<mybatis-spring-boot.version>2.1.2</mybatis-spring-boot.version>
<mysql.version>5.1.26</mysql.version>
<lombok.version>1.18.12</lombok.version>
<java.version>1.8</java.version>
</properties>
<dependencyManagement>
<dependencies>
<!-- spring-boot 依赖-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>${spring-boot.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<!-- spring-cloud 依赖-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>${spring-cloud.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<!-- spring-cloud-alibaba 依赖-->
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
<version>${spring-cloud-alibaba.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<!-- mybatis -->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>${mybatis-spring-boot.version}</version>
</dependency>
<!-- mysql -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>${mysql.version}</version>
</dependency>
<!-- test -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<version>${spring-boot.version}</version>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>
<!-- lombok -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>${lombok.version}</version>
</dependency>
</dependencies>
</dependencyManagement>
<!-- SpringBoot打包插件,可以将代码打包成一个可执行的jar包 -->
<build>
<finalName>${project.artifactId}</finalName>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>${java.version}</source>
<target>${java.version}</target>
</configuration>
</plugin>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<fork>true</fork>
<addResources>true</addResources>
</configuration>
</plugin>
</plugins>
</build>
</project>
3.3 服务注册中心
创建一个名称为spring-cloud-eureka的项目,作为服务注册中心。这里使用eureka 来管理服务的注册和发现,工程的pom文件如下:
<?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>
<parent>
<groupId>com.gallenzhang</groupId>
<artifactId>spring-cloud-parent</artifactId>
<version>1.0.0-SNAPSHOT</version>
<relativePath />
</parent>
<groupId>com.gallenzhang</groupId>
<artifactId>spring-cloud-eureka</artifactId>
<version>1.0.0-SNAPSHOT</version>
<name>spring-cloud-eureka</name>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
在启动类上加 @EnableEurekaServer表明这是个eureka sever。
/**
* <p>
* Eureka 启动类
* </p>
*
* @author gallenzhang
* @since 2020/6/20
**/
@SpringBootApplication
@EnableEurekaServer
public class SpringCloudEurekaApplication {
public static void main(String[] args) {
SpringApplication.run(SpringCloudEurekaApplication.class, args);
}
}
默认情况下eureka server 也是一个eureka client,所以eureka.client.register-with-eureka 和 eureka.client.fetch-registry属性都配置为false,表明自己是一个eureka server。配置文件application.yml 内容如下:
server:
port: 8000
spring:
application:
name: spring-cloud-eureka
eureka:
instance:
hostname: localhost
client:
#是否将自己注册到Eureka Server,默认为true
register-with-eureka: false
#是否从Eureka Server获取注册信息,默认为true
fetch-registry: false
serviceUrl:
#服务注册的URL
defaultZone: http://${eureka.instance.hostname}:${server.port}/eureka/
右键直接运行SpringCloudEurekaApplication类,启动完成后浏览器访问 http://127.0.0.1:8000/
看到这个页面表示 eureka 已经启动成功,这里的 No instances available
表示目前没有服务提供者注册上来。
3.4 服务提供者
创建一个名称为 spring-cloud-provider的项目,在pom文件中引入对应的依赖。
<?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>
<parent>
<groupId>com.gallenzhang</groupId>
<artifactId>spring-cloud-parent</artifactId>
<version>1.0.0-SNAPSHOT</version>
<relativePath />
</parent>
<groupId>com.gallenzhang</groupId>
<artifactId>spring-cloud-provider</artifactId>
<version>1.0.0-SNAPSHOT</version>
<name>spring-cloud-provider</name>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
配置文件application.yml 内容如下:
server:
port: 8100
spring:
application:
name: spring-cloud-provider #服务之间的调用都是根据这个 name
eureka:
client:
serviceUrl:
defaultZone: http://localhost:8000/eureka/
为了方便测试,项目新建一个HelloController 类
/**
* <p>
* Hello Controller
* </p>
*
* @author gallenzhang
* @since 2020/6/20
**/
@RestController
public class HelloController {
@Value("${server.port}")
private String port;
@RequestMapping("/hello")
public String sayHello(@RequestParam(value = "name", defaultValue = "gallenzhang") String name){
return "hello " + name + " , I am from port:" + port;
}
}
启动类添加@EnableEurekaClient 注解
/**
* <p>
* 提供者启动类
* </p>
*
* @author gallenzhang
* @since 2020/6/20
**/
@SpringBootApplication
@EnableEurekaClient
public class SpringCloudProviderApplication {
public static void main(String[] args) {
SpringApplication.run(SpringCloudProviderApplication.class, args);
}
}
右键运行SpringCloudProviderApplication,将提供者注册到eureka上去。此时访问 http://127.0.0.1:8000/
服务提供者已经注册到 eureka 上了,服务名称就是 spring.application.name 值的大写。
浏览器访问 http://localhost:8100/hello?name=allen ,在浏览器上可以看到响应结果。
hello allen , I am from port:8100
3.5 服务消费者
Feign 是一个声明式的伪Http客户端,使用它只要创建一个接口并添加注解@FeignClient 即可,它使得写Http客户端变得简单。Feign默认集成了Ribbon,并和 eureka 结合,实现了客户端负载均衡。
创建一个名称为 spring-cloud-consumer的项目,在pom文件中引入对应的依赖。
<?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>
<parent>
<groupId>com.gallenzhang</groupId>
<artifactId>spring-cloud-parent</artifactId>
<version>1.0.0-SNAPSHOT</version>
<relativePath />
</parent>
<groupId>com.gallenzhang</groupId>
<artifactId>spring-cloud-consumer</artifactId>
<version>1.0.0-SNAPSHOT</version>
<name>spring-cloud-consumer</name>
<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-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
配置文件application.yml 内容如下:
server:
port: 8200
spring:
application:
name: spring-cloud-consumer #服务之间的调用都是根据这个name
eureka:
client:
serviceUrl:
defaultZone: http://localhost:8000/eureka/
在程序的启动类SpringCloudConsumerApplication ,加上@EnableFeignClients注解开启 Feign 的功能。
/**
* <p>
* 消费者启动类
* </p>
*
* @author gallenzhang
* @since 2020/6/20
**/
@SpringBootApplication
@EnableEurekaClient
@EnableDiscoveryClient
@EnableFeignClients
public class SpringCloudConsumerApplication {
public static void main(String[] args) {
SpringApplication.run(SpringCloudConsumerApplication.class, args);
}
}
定义一个接口,通过@FeignClient 来指定调用哪个服务。这里指定了调用 spring-cloud-provider 服务的 hello
接口。
/**
* <p>
* Hello Client
* </p>
*
* @author gallenzhang
* @since 2020/6/20
**/
@FeignClient(value = "spring-cloud-provider")
public interface HelloClient {
/**
* 客户端调用服务
* @param name
* @return
*/
@RequestMapping(value = "/hello")
String sayHelloFromClient(@RequestParam(value = "name") String name);
}
在 Controller 层对外暴露一个 /hello
的Api接口,Controller中通过上面定义的HelloClient来消费服务。
/**
* <p>
* Hello Controller
* </p>
*
* @author gallenzhang
* @since 2020/6/20
**/
@RestController
public class HelloController {
@Autowired
private HelloClient helloClient;
@RequestMapping(value = "/hello")
public String sayHello(@RequestParam String name) {
return helloClient.sayHelloFromClient(name);
}
}
启动 spring-cloud-eureka 服务。 修改 spring-cloud-provider 项目 application.yml文件中 server.port值,分别启动三次,端口号依次为 8100、8101、8102。
然后右键运行SpringCloudConsumerApplication,启动消费者服务。
浏览器访问 http://127.0.0.1:8200/hello?name=allen 浏览器会交替输出。
hello allen , I am from port:8100
hello allen , I am from port:8101
hello allen , I am from port:8102
源码下载:https://github.com/gallenzhang/spring-cloud-example
四、学习资料
SpringCloud 官网
SpringCloud 中文网
SpringCloud 开发指南-英文版
本文主要参考如下两篇文章。
Spring Cloud 快速入门
史上最简单的 SpringCloud 教程 | 终章
如果想进一步讨论SpringCloud 或者 其他技术问题,欢迎加我微信一起交流(请注明来意):1298552064