首先介绍一下单体架构与微服务架构:
单体架构:
data:image/s3,"s3://crabby-images/57a1e/57a1e2fedb0ad241c6ee6ecae74ad4f905e5e5bf" alt=""
微服务:
data:image/s3,"s3://crabby-images/84397/843979502cfb905eedb7df0478de45db0409565b" alt=""
SpringCloud:
data:image/s3,"s3://crabby-images/06d64/06d64412ebcc20d63f9636272e02aa3ffda43825" alt=""
版本:
data:image/s3,"s3://crabby-images/02fee/02fee3c001c7f95a6d4c11e80d93641017fa93f2" alt=""
标黑部分为目前企业使用最多的版本,因为它支持jdk8、jdk11,下面使用SpringCloud也会使用这个版本。
服务拆分:
拆分原则:
什么时候拆分?
data:image/s3,"s3://crabby-images/2c323/2c32392e281c6b9f660e8e85dd8780998e0283cf" alt=""
怎么拆分?
data:image/s3,"s3://crabby-images/231dc/231dc2f6f84e8d7e8afb8f3c26dd73d3793e1fc6" alt=""
工程结构:
拆分后的工程结构有两种:
1.独立Project,将拆分后的所有服务放到一个文件夹中,适合大型项目(有很多微服务)。
2.Maven聚合,创建一个Project,然后在其下面创建Module为微服务模块。
服务拆分后,不同微服务之间可能有调用,比如购物车服务中会调用到商品服务,那么需要进行远程调用。
远程调用:
1.方法一:(不推荐)
data:image/s3,"s3://crabby-images/8632a/8632a2db6988621fd3f888c607d5ceca386a8ace" alt=""
就可以利用RestTemplate在购物车服务中向商品服务发送http请求并获取响应体。
问题:
为了减小服务压力,http请求接收的服务可能会部署多个,所以并不能确定向哪个服务发送http请求。
2.方法二:
注册中心:
原理:
data:image/s3,"s3://crabby-images/881b5/881b50e3c7073af98cef203543020182ec225e18" alt=""
data:image/s3,"s3://crabby-images/9e46a/9e46a6280008a8d098345f604b5f12a6022694f9" alt=""
Nacos注册中心:
Nacos是目前国内企业中占比最多的注册中心组件。它是阿里巴巴的产品,目前以及假如SpringCloudAlibaba中。
使用时需要首先进行数据库中nacos数据库创建,并且向数据库中导入信息,并启动nacos镜像。
服务注册:
data:image/s3,"s3://crabby-images/9f0be/9f0bebd87e43f548522605013a154c9b0028cfe1" alt=""
服务发现:
data:image/s3,"s3://crabby-images/04099/04099a1ad58d3e60719443b7fac2ef1267fa74da" alt=""
3.OpenFeign(推荐):
data:image/s3,"s3://crabby-images/32bcd/32bcdbd6803508d9d19f540a290a92b7f028be1f" alt=""
使用步骤:
data:image/s3,"s3://crabby-images/1485b/1485bd76e8bef9c888649897481d68fb89d7b187" alt=""
data:image/s3,"s3://crabby-images/7c53b/7c53bcacf2454a21e99e0b61d96c4d7383601f9b" alt=""
注意:因为OpenFeign底层发送http请求是通过Client发送的,而Client每一次发送都需要重新创建连接,所以效率很低,因为我们使用连接池优化
连接池:
data:image/s3,"s3://crabby-images/6dd5a/6dd5a82478ba6bdb71524a4030016b32d333c356" alt=""
连接池使用:
data:image/s3,"s3://crabby-images/74e5a/74e5a69aed78090ea08875fff9bf91121def538d" alt=""
实践方案:
方案一:(较推荐)
data:image/s3,"s3://crabby-images/e1ce7/e1ce7bf2a15ede99ffaec81a1d63885714143df1" alt=""
特点:
代码结构更合理,耦合度非常低,但是项目结构变复杂。
方案二:
data:image/s3,"s3://crabby-images/5d3bc/5d3bc2916a533190156f41ac4ab2ce9ee61376cd" alt=""
特点:
结构更简单,使用更方便,但代码耦合度更高一点。
定义的FeignClient不在扫描包范围时:
data:image/s3,"s3://crabby-images/fd159/fd159b0de449a053dee62dd2da520bdfc7f84c1a" alt=""
日志:
data:image/s3,"s3://crabby-images/8ddf5/8ddf5f9f5b6e8e3300f51d36c8bc4396b2fe9aa4" alt=""
data:image/s3,"s3://crabby-images/82fa4/82fa419ac44eac3e6fb03725e1d0d2066fb59b41" alt=""
网关:
网关就是网络的关口,负责请求的路由、转发、身份校验。
data:image/s3,"s3://crabby-images/3b683/3b6834f2d766b4d4735a092c3d31cf7dde9d6063" alt=""
SpringCloud中的网关的实现:
data:image/s3,"s3://crabby-images/c0ce6/c0ce629a0da13fac06118891eac06b477a7c4e66" alt=""
这里我们使用Spring Cloud Gateway。
快速入门
data:image/s3,"s3://crabby-images/af4f9/af4f9ac3e1180d95770201bcd8bddaee20e73be4" alt=""
路由属性:
路由断言:
data:image/s3,"s3://crabby-images/6abde/6abde7981df04eaeac185f09bc0350f2f643f308" alt=""
路由过滤器:
data:image/s3,"s3://crabby-images/9d192/9d1923d9e1175632bce08dfb85a0a58d0fcbd17b" alt=""
如果想要给所有服务都配置一种路由过滤器,可以在与routes同级的位置配置default-filters,然后输入要配置的路由过滤器。
网关请求处理流程:
data:image/s3,"s3://crabby-images/b33bf/b33bf3ab8cf1776f8ee38bd9ff427a1ca97cf670" alt=""
因此,我们要在网关内进行登录校验,需要自定义pre的过滤器进行jwt校验。
自定义过滤器:
data:image/s3,"s3://crabby-images/4de43/4de43b83b5be4d309c36dde9b0d460ec671eaa6a" alt=""
自定义GlobalFilter:(大多数情况)
参数:
data:image/s3,"s3://crabby-images/63a8b/63a8b43e92d45d19b452610dca40bc03580091d8" alt=""
步骤:
data:image/s3,"s3://crabby-images/fd00b/fd00bebd0b99bd83350abbeaeb82123e5dceb3ba" alt=""
可以实现Ordered接口,就能够定义过滤器优先级。
自定义GatewayFilter
data:image/s3,"s3://crabby-images/1f890/1f890d5659c699e871dc5995d3a64dca2762881c" alt=""
实现登录校验:
我们可以在网关中通过自定义过滤器实现登录校验
实现网关传递用户信息:
data:image/s3,"s3://crabby-images/c9025/c902592c417ff055dd36e03741fcb70a6e7600e8" alt=""
网关保存用户到请求头:
data:image/s3,"s3://crabby-images/126b5/126b54f658e5de535140534a70999e2fef0a0509" alt=""
完整代码:
java
@Component
@RequiredArgsConstructor
public class AuthGlobalFilter implements GlobalFilter, Ordered {
private final AuthProperties authProperties;
private final JwtTool jwtTool;
private final AntPathMatcher antPathMatcher = new AntPathMatcher();
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
//1.获取request
ServerHttpRequest request = exchange.getRequest();
//2.判断是否需要做登录拦截
String path = request.getPath().toString();
if (isExclude(path)) {
return chain.filter(exchange);
}
//3.获取token
String token = null;
HttpHeaders headers = request.getHeaders();
List<String> authorization = headers.get("Authorization");
if (authorization != null && authorization.size() > 0) {
token = authorization.get(0);
}
//4.校验并解析token
Long userId = null;
try {
userId = jwtTool.parseToken(token);
} catch (UnauthorizedException e) {
ServerHttpResponse response = exchange.getResponse();
response.setStatusCode(HttpStatus.UNAUTHORIZED);
return response.setComplete();
}
//5.传递用户信息
String userInfo = userId.toString();
ServerWebExchange swe = exchange.mutate()
.request(builder -> builder.header("user-info", userInfo))
.build();
//6.放行
return chain.filter(swe);
}
private boolean isExclude(String path) {
List<String> excludePaths = authProperties.getExcludePaths();
for (String pathPattern : excludePaths) {
if (antPathMatcher.match(pathPattern, path)) {
return true;
}
}
return false;
}
@Override
public int getOrder() {
return 0;
}
}
antPathMather是一个路径字符匹配器API,它可以方便我们对路径进行匹配。
编写SpringMVC拦截器
编写拦截器以获取登录用户
data:image/s3,"s3://crabby-images/75447/7544776bede7d3bc5b5c631bacdff439f9b76af9" alt=""
步骤:
1.首先编写拦截器类:
需要让拦截器实现HandlerInterceptor接口,重写preHandle和afterCompletion方法,preHandle是在传给后续微服务前执行,所以在这个方法中将用户信息存入ThreadLocal。afterCompletion是在使用完后执行,所以删除存储的用户信息,防止内存泄漏。
java
public class UserInfoInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
//1.获取用户信息
String userInfo = request.getHeader("user-info");
//2.判断是否为空,不为空则存入ThreadLocal
if(StrUtil.isNotBlank(userInfo)) {
UserContext.setUser(Long.parseLong(userInfo));
}
//3.放行
return true;
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
UserContext.removeUser();
}
}
2.编写配置类
编写完拦截器后,需要在编写配置类使之生效
需要注意的有
**·**配置类注解@Configuration表名它是一个配置类
**·**注解@ConditionalOnClass,是为了让网关不接收这个拦截器,让其他微服务接收,注解实现让具有DispatcherServlet.class的微服务接收拦截器,这是SpringMVC特有的class对象,网关中没有配置SpringMVC所以网关就不会接收。
**·**因为是SpringMVC中的拦截器,配置类要继承WebMvcConfigurer接口,并实现addInterceptors方法,添加刚刚编写的拦截器。
java
@Configuration
@ConditionalOnClass(DispatcherServlet.class)
public class MvcConfig implements WebMvcConfigurer {
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new UserInfoInterceptor());
}
}
3.将配置类放入SpringBoot配置文件中:
将配置类全类名放入spring.factories配置文件中
data:image/s3,"s3://crabby-images/6eaf9/6eaf947a7080fd5cd019952b9701bcfb5ad84ea5" alt=""
实现微服务之间传递用户信息:
可以使用OpenFeign实现
data:image/s3,"s3://crabby-images/88647/886473f5e0ae70a8689ccdc82d9c1448a45508ff" alt=""
在Feign的配置类中直接定义拦截器:
java
public class DefaultFeignConfig {
@Bean
public Logger.Level feignLoggerLevel() {
return Logger.Level.FULL;
}
@Bean
public RequestInterceptor userInfoRequestInterceptor() {
return new RequestInterceptor() {
@Override
public void apply(RequestTemplate requestTemplate) {
Long userId = UserContext.getUser();
if (userId != null) {
requestTemplate.header("user-info", userId.toString());
}
}
};
}
}
注意微服务启动类上要有注解:
@EnableFeignClients(basePackages = "com.hmall.api.client", defaultConfiguration = DefaultFeignConfig.class)
总结:
data:image/s3,"s3://crabby-images/b5d9e/b5d9e0e479b862d51f9e61ce8dec750f38b8920c" alt=""
配置管理:
共享配置:
为什么要使用配置管理:
data:image/s3,"s3://crabby-images/1ff12/1ff12435db199370509672b316552483e3f70b2e" alt=""
使用配置管理服务:
data:image/s3,"s3://crabby-images/5f39d/5f39d785650611244a8649c224b204d4769f1d49" alt=""
我们在服务远程调用时使用的注册中心Nacos,就有配置管理服务的功能
1.添加配置到Nacos:
data:image/s3,"s3://crabby-images/cafaa/cafaaf2dc89bb9cc88cbf11dfadbfb1eafe5a7a7" alt=""
添加微服务中共享的那部分配置即可。
注意可以使用变量,即${配置文件中的路径},在原yml配置文件中配置变量,即可正常读取。
2.拉取共享配置
data:image/s3,"s3://crabby-images/c1a01/c1a01710fdcd2cb7151f1bd9eb32cc1b27a206af" alt=""
①引入依赖
data:image/s3,"s3://crabby-images/19267/192676fd63d75aa7a879534eb3673826417bceb7" alt=""
②新建bootstrap.yaml
在微服务中创建bootstrap.yaml配置文件,在bootstrap.yaml中配置的信息不需要再配置了。
data:image/s3,"s3://crabby-images/1557e/1557e4827b4f49cbe9c681eb074f02df54ea808d" alt=""
配置热更新:
配置热更新:当修改配置文件中的配置时,微服务无需重启即可使配置生效。
前提条件:
data:image/s3,"s3://crabby-images/83bc8/83bc8fce40147c4daa7b3ecaee2ccdbd8ddad475" alt=""
步骤
1.在nacos中定义一个与微服务名有关的配置文件
data:image/s3,"s3://crabby-images/1557e/1557e4827b4f49cbe9c681eb074f02df54ea808d" alt=""
在上述bootstrap.yaml中已经有了这部分信息
2.加载属性
一般采用这种方式:
data:image/s3,"s3://crabby-images/76eb2/76eb2936a0034af205e7adb9a84c6e6d5528a1fe" alt=""
完成后,在nacos中配置一旦变更,就会实时更新。
服务保护和分布式事务:
雪崩问题:
雪崩问题是服务保护方面经常碰到的一个问题,即:
微服务中调用链路中的某个服务故障,引起整个链路中的所有微服务都不可用,这就是雪崩
产生原因:
data:image/s3,"s3://crabby-images/e0927/e09270ffc65aabdd18546164639e27b9484bb507" alt=""
解决思路:
data:image/s3,"s3://crabby-images/99ded/99ded073ad10d28665e96bdff2dea76c5b8afaec" alt=""
服务保护方案:
请求限流:
data:image/s3,"s3://crabby-images/b0fc1/b0fc1cb2c84a752e9713b98108d734da2a9449de" alt=""
线程隔离:
data:image/s3,"s3://crabby-images/24baa/24baa14d54fa6b74e541aeba3ac383be8c5f5df1" alt=""
服务熔断:
data:image/s3,"s3://crabby-images/886ab/886ab81b3c778dfc74373793e7dc5b81a4a99495" alt=""
解决方案总结:
data:image/s3,"s3://crabby-images/ab512/ab512af6b7ef35090d15db2954551beef9bc8449" alt=""
服务保护技术:
我们可以使用服务保护技术方便我们完成上述解决方案。
data:image/s3,"s3://crabby-images/02ca7/02ca7991ddb4a5ecc0c6679bde1b168ab59052f9" alt=""
Sentinel:
Sentinel是阿里巴巴开源的一款微服务流量控制组件。官网地址:home | Sentinel (sentinelguard.io)
簇点链路介绍:
data:image/s3,"s3://crabby-images/fb25c/fb25c05d32dbd2f79cfd503abf1286e76d33accd" alt=""
使用方法:
引入依赖:
XML
<!--sentinel-->
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-sentinel</artifactId>
</dependency>
配置控制台:
XML
spring
cloud:
sentinel:
transport:
dashboard: localhost:8090
http-method-specify: true #是否设置请求方式作为资源名称
请求限流:
在簇点链路后面点击流控按钮,即可对其做限流配置:
data:image/s3,"s3://crabby-images/376b9/376b9b960dd668de41875b6b6744b5f057e679d3" alt=""
点击后弹出下面的窗口:
data:image/s3,"s3://crabby-images/09eaf/09eaf3fdfa1191db274ff5862fed3348867b7a54" alt=""
阈值类型默认为QPS,即每秒请求的数量,在单机阈值处填写数据点击新增即可。
线程隔离:
线程隔离也是要点击流控按钮,在窗口中配置并发线程数
data:image/s3,"s3://crabby-images/8a2a8/8a2a8b7fe4e972a90eda5925be6a389d1286b415" alt=""
Fallback:
实现Fallback要对FeignClient操作,所以需要让它成为Sentinel的簇点资源。
data:image/s3,"s3://crabby-images/b3ef2/b3ef29f1161939e88966c0153ee5b6a7e912eeb3" alt=""
编写步骤:
data:image/s3,"s3://crabby-images/2a9d3/2a9d3064f76f2c481175e0a7aeb2c92f9d12aee5" alt=""
步骤一:
在编写的cilent包下新建fallback包再新建FallbackFactory类即可。
步骤二:
data:image/s3,"s3://crabby-images/15bc0/15bc0dd6bef34e6ebe530af7f215c4eb92dfa99d" alt=""
步骤三:
data:image/s3,"s3://crabby-images/30b19/30b1958fa748ff869cbe478fae620e82288326ae" alt=""
服务熔断:
服务熔断通过断路器实现。
断路器原理:
data:image/s3,"s3://crabby-images/6810a/6810a4f5ff6732795bf550b787c772192d16a35f" alt=""
使用方法:
点击簇点链路的熔断按钮,弹出下面窗口,默认选取的是慢调用比例,最大RT(response time)即最大响应时间,超过整个响应时间的请求被归为慢调用,比例阈值就是当慢调用的请求的比例超过比例阈值时,就会进行熔断,最小请求数量就是要对这么多次的请求一起判断,统计时长就是统计的周期。
data:image/s3,"s3://crabby-images/3270f/3270f0e6de28bb3218b0c4ac530686606dc24d9d" alt=""
分布式事务:
分布式事务调用了其他服务,举例:
data:image/s3,"s3://crabby-images/98605/9860516d051e11a803d64cfd6a7f35412e9e8bb9" alt=""
如果不解决分布式事务,当程序正常进行,但是到第三步扣减商品库存出现问题,比如库存不足报错,这时库存没有正常扣减,但是购物车已经被清除,没有保证原子性。
注意这种情况不能使用@Transactional注解,它只适用于单个服务。
解决思路:
各个子事务之间必须能感知到批次的事务状态,才能保证状态一致。
data:image/s3,"s3://crabby-images/bbb3b/bbb3b163704fe8d8ab659da8204d909c53690855" alt=""
Seata:
data:image/s3,"s3://crabby-images/51d14/51d145e4e4b1c6ce817f9e46a5f68a1d1ce4373b" alt=""
它本身也是一个微服务。
架构:
data:image/s3,"s3://crabby-images/7b1c2/7b1c2e59e9eedcc300576292319f3c94b961970c" alt=""
使用步骤:
建表:
Seata支持多种存储模式,但考虑到持久化的需要,我们一般选择基于数据库存储。
data:image/s3,"s3://crabby-images/4c2f5/4c2f57bfa346b4d6a7b336c172ba24bb70fdaffb" alt=""
准备配置文件:
将seata的配置文件放入服务器或虚拟机/root目录下
用docker部署:
需要注意,要确保nacos、mysql都在hm-net网络中。
微服务整合Seata:
首先需要引入依赖:
XML
<!--统一配置管理-->
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
</dependency>
<!--读取bootstrap文件-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-bootstrap</artifactId>
</dependency>
<!--seata-->
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-seata</artifactId>
</dependency>
data:image/s3,"s3://crabby-images/42e4e/42e4eb591b59a455234f667f2fb64ad21ffd5599" alt=""
我将它定义在了nacos共享配置中。
最后,新建bootstrap.yaml文件,定义如下配置
XML
spring:
application:
name: ****-service # 服务名称
profiles:
active: dev
cloud:
nacos:
server-addr: ***.***.***.*** # nacos地址
config:
file-extension: yaml # 文件后缀名
shared-configs: # 共享配置
- dataId: shared-seata.yaml # 共享seata配置
Seata模式:
XA模式:
data:image/s3,"s3://crabby-images/fed49/fed491c74346bebb68dd815db5bec43d1f23c924" alt=""
优点:
data:image/s3,"s3://crabby-images/db2ff/db2ff31f8c73375bd3caa01e39a543aface434d5" alt=""
缺点:
data:image/s3,"s3://crabby-images/2b997/2b9971098e9142ab9886ba4ff897a6eeea3fda98" alt=""
实现:
data:image/s3,"s3://crabby-images/9d5be/9d5be8eb71ab217dc210d9aeeb1004d2f625d194" alt=""
AT模式:
data:image/s3,"s3://crabby-images/14730/14730c863805cff7a0d7704ada1e6909be4397ab" alt=""
实现:
data:image/s3,"s3://crabby-images/a18e4/a18e43cf54bdd934a7eb203815e7787151c93c7f" alt=""
XA模式与AT模式的区别:
data:image/s3,"s3://crabby-images/87fa2/87fa28ef305ad3c13a138cdfc69fd0ea1ada5b24" alt=""
(即XA模式在整个过程中数据库中的信息都是一致的,而AT模式在一阶段提交完成后,有服务出现问题,在二阶段根据数据快照恢复数据前,会出现短暂的数据不一致情况)
后续学习:
RabbitMQ:
详见作者的下一篇文章:Java_RabbitMQ
Elasticsearch(ES):
详见作者的下下篇文章:Java_Elasticsearch(ES)