Spring Cloud知识点全总结(七)

简介: Spring Cloud知识点全总结

(6)优化


实际部署时,需要给做反向代理的nginx服务器设置一个域名,这样后续如果有服务器迁移nacos的客户端也无需更改配置.


Nacos的各个节点应该部署到多个不同服务器,做好容灾和隔离


这一小节的一些坑:


cluster.conf里面要填写本机的真实IP,不能写127.0.0.1 。


单机启动nacos服务后,服务注册出现以下异常:


c8e539d9fae94e1cb236b86eb99ae2bc.png


解决办法:

删除data目录下的protocol文件夹,重启服务即可。

异常原因:

1.4.0使用了jraft, jraft会记录前一次启动的集群地址,如果重启机器ip变了的话,会导致jraft记录的地址失效,从而导致选主出问题。

1.4.0之后,单机情况下也是存在节点了。流程和集群一样,需要先选出leader,再提供服务。


配置好Nacos集群,我们重启Nacos后,我们之前的Nacos配置文件会被清掉一次,所以最好把之前项目里面读取Nacos配置文件的代码都删除掉。


七. Feign远程调用


这一章会先分析一下RestTemplate存在的问题,然后学习用Feign去替代RestTemplate。当然,我们还会学习一下Feign的自定义的一些配置,以及使用时的一些性能优化。最后会学习Feign在企业当中的最佳实践方案。


1. RestTemplate存在的问题


先来看我们以前利用RestTemplate发起远程调用的代码:


f0149e2f3df144ad9c93084611fd9e2f.png


这个请求是通过URL地址,指明要访问的服务名称,还有请求路径,以及请求的参数信息。而后由RestTemplate帮我们向指定地址发起请求,再把结果转换成对应类型。


这段代码存在以下的问题:


•代码可读性差,编程体验不统一


•参数复杂URL难以维护


Feign是一个声明式的http客户端,官方地址:https://github.com/OpenFeign/feign


其作用就是帮助我们优雅的实现http请求的发送,解决上面提到的问题。


742e801647fe45a99382bfdbd3c81e5a.png


2. Feign替代RestTemplate


Feign的使用步骤如下:


(1) 引入依赖


我们在order-service服务的pom文件中引入feign的依赖:

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>


(2) 添加注解

在order-service的启动类添加注解开启Feign的功能:


@EnableFeignClients
@MapperScan("com.haiexijun.order.mapper")
@SpringBootApplication
public class OrderApplication {
    public static void main(String[] args) {
        SpringApplication.run(OrderApplication.class, args);
    }
}


(3) 编写Feign的客户端


在order-service中新建一个接口,内容如下:


package com.haiexijun.order.client;
import com.haiexijun.order.pojo.User;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
@FeignClient("userservice")
public interface UserClient {
    @GetMapping("/user/{id}")
    User findById(@PathVariable("id") Long id);
}


这个客户端主要是基于SpringMVC的注解来声明远程调用的信息,比如:


服务名称:userservice

请求方式:GET

请求路径:/user/{id}

请求参数:Long id

返回值类型:User

这样,Feign就可以帮助我们发送http请求,无需自己使用RestTemplate来发送了。


(4)测试


修改order-service中的OrderService类中的queryOrderById方法,使用Feign客户端代替RestTemplate:


ff856eeae96846718a635de9402bb6da.png


是不是看起来优雅多了。而且Feign非常强大,不仅实现了远程调用,还实现了负载均衡。


(5) 总结

使用Feign的步骤:


① 引入依赖


② 添加@EnableFeignClients注解


③ 编写FeignClient接口


④ 使用FeignClient中定义的方法代替RestTemplate


2. 自定义配置Feign


SpringBoot虽然帮我们实现了自动装配,但它是允许我们覆盖默认配置的。


Feign可以支持很多的自定义配置,如下表所示(只是部分):


image.png


一般情况下,默认值就能满足我们使用,如果要自定义时,只需要创建自定义的@Bean覆盖默认Bean即可。

有配置文件和java代码两种方式。

方式一配置文件方式

基于配置文件修改feign的日志级别可以针对单个服务:


feign:  
  client:
    config: 
      userservice: # 针对某个微服务的配置
        loggerLevel: FULL #  日志级别 


也可以针对所有服务:


feign:  
  client:
    config: 
      default: # 这里用default就是全局配置,如果是写服务名称,则是针对某个微服务的配置
        loggerLevel: FULL #  日志级别



而日志的级别分为四种:


NONE:不记录任何日志信息,这是默认值。

BASIC:仅记录请求的方法,URL以及响应状态码和执行时间

HEADERS:在BASIC的基础上,额外记录了请求和响应的头信息

FULL:记录所有请求和响应的明细,包括头信息、请求体、元数据。

方式二Java代码方式

也可以基于Java代码来修改日志级别,先声明一个类,然后声明一个Logger.Level的对象:


public class DefaultFeignConfiguration  {
    @Bean
    public Logger.Level feignLogLevel(){
        return Logger.Level.BASIC; // 日志级别为BASIC
    }
}


如果要全局生效,将其放到启动类的@EnableFeignClients这个注解中:


@EnableFeignClients(defaultConfiguration = DefaultFeignConfiguration .class)


如果是局部生效,则把它放到对应某一个服务的@FeignClient这个注解中:


@FeignClient(value = "userservice", configuration = DefaultFeignConfiguration .class)


3. Feign使用优化


Feign的性能已经很好了,但是还是有优化的余地。我们先来了解一下Feign底层的实现:


Feign底层发起http请求,依赖于其它的框架。其底层客户端实现包括:


•URLConnection:默认实现(JDK自带的),不支持连接池


•Apache HttpClient :支持连接池


•OKHttp:支持连接池


因此提高Feign的性能主要手段就是使用连接池代替默认的URLConnection。


这里我们用Apache的HttpClient来演示。


(1) 引入依赖


在order-service的pom文件中引入Apache的HttpClient依赖:


<!--httpClient的依赖 -->
<dependency>
    <groupId>io.github.openfeign</groupId>
    <artifactId>feign-httpclient</artifactId>
</dependency>


(2) 配置连接池

在order-service的application.yml中添加配置:


feign:
  client:
    config:
      default: # default全局的配置
        loggerLevel: BASIC # 日志级别,BASIC就是基本的请求和响应信息
  httpclient:
    enabled: true # 开启feign对HttpClient的支持
    max-connections: 200 # 最大的连接数
    max-connections-per-route: 50 # 每个路径的最大连接数


接下来,在FeignClientFactoryBean中的loadBalance方法中打断点:


a683a2cbc3bc495f833194ffa01742c4.png


Debug方式启动order-service服务,可以看到这里的client,底层就是Apache HttpClient:


23de08fd50d94f4f9dbe5b983142a19a.png


总结,Feign的优化:


1.日志级别尽量用basic


2.使用HttpClient或OKHttp代替URLConnection


① 引入feign-httpClient依赖


② 配置文件开启httpClient功能,设置连接池参数


4. Feign最佳实践的分析


什么是最佳实践呢?就是企业在使用一个东西的过程中,各种踩坑,最后总结出来的一个比较好的使用方式。


观察可以发现,Feign的客户端与服务提供者的controller代码非常相似:


feign客户端:


cf1420ce5dc944b6bbcdb478256222c7.png


UserController:


14df1cea574e4bbdb4f059201d5da536.png


有没有一种办法简化这种重复的代码编写呢?


这一节会介绍两种比较好的Feign的实践方案。


方式一:继承方式


一样的代码可以通过继承来共享:


1)定义一个API接口,利用定义方法,并基于SpringMVC注解做声明。


2)Feign客户端和Controller都集成改接口


56138e0930204150ac364b2d849ea572.png


优点:


简单

实现了代码共享

缺点:


服务提供方、服务消费方紧耦合


参数列表中的注解映射并不会继承,因此Controller中必须再次声明方法、参数列表、注解


方式二:抽取方式

将Feign的Client抽取为独立模块,并且把接口有关的POJO、默认的Feign配置都放到这个模块中,提供给所有消费者使用。


例如,将UserClient、User、Feign的默认配置都抽取到一个feign-api包中,所有微服务引用该依赖包,即可直接使用。


650f4d8b160449b393046f49e139b79e.png


5. Feign 最佳实践的代码实现

下面以抽取的方式来实现Feign的最佳实践。


(1) 抽取


首先创建一个module,命名为feign-api:


8bd5dfb6b6604705a053c8adc0f12e11.png


项目结构:


9b12c7ca8e52456a9fba492414ab4b79.png


在feign-api中然后引入feign的starter依赖:


<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>


然后,order-service中编写的UserClient、User、DefaultFeignConfiguration都复制到feign-api项目中。


(2)在order-service中使用feign-api


首先,删除order-service中的UserClient、User、DefaultFeignConfiguration等类或接口。


在order-service的pom文件中中引入feign-api的依赖:


<dependency>
    <groupId>com.haiexijun.demo</groupId>
    <artifactId>feign-api</artifactId>
    <version>1.0</version>
</dependency>


修改order-service中的所有与上述三个组件有关的导包部分,改成导入feign-api中的包

(3) 重启测试

(4) 解决扫描包问题

方式一:

指定Feign应该扫描的包:


@EnableFeignClients(basePackages = "com.haiexijun.feign.clients")


方式二:

指定需要加载的Client接口:


@EnableFeignClients(clients = {UserClient.class})


八.Gateway服务网关


Spring Cloud Gateway 是 Spring Cloud 的一个全新项目,该项目是基于 Spring 5.0,Spring Boot 2.0 和 Project Reactor 等响应式编程和事件流技术开发的网关,它旨在为微服务架构提供一种简单有效的统一的 API 路由管理方式。


1. 为什么需要网关


Gateway网关是我们服务的守门神,所有微服务的统一入口。


我们的微服务如果直接任人都能发请求来访问是不是不太安全啊。你要知道,不是所有的业务都是对外公开的,有好多业务属于公司内部的或者管理人员才可以去访问的,所以得对用户的身份做一个认证,如果说是我们的内部人员,才允许访问一些服务。网关就是来做这样一件事情的。


网关的核心功能特性:


请求路由

权限控制

限流

架构图:


9e393f98a27a4e6cadb067c3aa2c9900.png


权限控制:网关作为微服务入口,需要校验用户是是否有请求资格,如果没有则进行拦截。


路由和负载均衡:一切请求都必须先经过gateway,但网关不处理业务,而是根据某种规则,把请求转发到某个微服务,这个过程叫做路由。当然路由的目标服务有多个时,还需要做负载均衡。


限流:当请求流量过高时,在网关中按照微服务能够接受的速度来放行请求,避免服务压力过大。


在SpringCloud中网关的实现包括两种:


gateway

zuul

Zuul是基于Servlet的实现,属于阻塞式编程。而Spring Cloud Gateway则是基于Spring5中提供的WebFlux,属于响应式编程的实现,具备更好的性能。所以我们选择Spring Cloud Gateway。


2. Gateway快速入门


下面,我们就演示下网关的基本路由功能。基本步骤如下:


创建SpringBoot工程gateway,引入网关依赖

编写启动类

编写基础配置和路由规则

启动网关服务进行测试


(1)创建gateway服务,引入依赖


创建项目


6a870329557641389e56644aefd41c5b.png


引入依赖:


<!--网关-->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-gateway</artifactId>
</dependency>
<!--nacos服务发现依赖-->
<dependency>
    <groupId>com.alibaba.cloud</groupId>
    <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>


(2)编写启动类


package com.haiexijun.gateway;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class GatewayApplication {
  public static void main(String[] args) {
    SpringApplication.run(GatewayApplication.class, args);
  }
}


(3)编写基础配置和路由规则

创建application.yml文件,内容如下:


server:
  port: 10010 # 网关端口
spring:
  application:
    name: gateway # 服务名称
  cloud:
    nacos:
      server-addr: localhost:8848 # nacos地址
    gateway:
      routes: # 网关路由配置
        - id: user-service # 路由id,自定义,只要唯一即可
          # uri: http://127.0.0.1:8081 # 路由的目标地址 http就是固定地址
          uri: lb://userservice # 路由的目标地址 lb就是负载均衡,后面跟服务名称
          predicates: # 路由断言,也就是判断请求是否符合路由规则的条件
            - Path=/user/** # 这个是按照路径匹配,只要以/user/开头就符合要求
         - id: order-service
          uri: lb://orderservice
          predicate:
            - Path=/order/**


我们将符合Path 规则的一切请求,都代理到 uri参数指定的地址。


本例中,我们将 /user/**开头的请求,代理到lb://userservice,lb是负载均衡,根据服务名拉取服务列表,实现负载均衡。


(4)重启测试


重启网关,访问http://localhost:10010/order/101时,符合/user/**规则,请求转发到uri:http://userservice/order/101,得到了结果:


a0724cff0a0d40c3a5b466025cb8348b.png


(5)网关路由的流程图

整个访问的流程如下:


8f70515184834e639f5238cd03fed669.png

总结:

网关搭建步骤:

创建项目,引入nacos服务发现和gateway依赖

配置application.yml,包括服务基本信息、nacos地址、路由

接下来,就重点来学习路由断言和路由过滤器的详细知识


3.路由断言工厂


路由断言工厂 Route Predicate Factory


网关路由可以配置的内容包括:


路由id:路由的唯一标示


路由目标(uri):路由的目标地址,http代表固定地址,lb代表根据服务名负载均衡


路由断言(predicates):判断路由的规则,


路由过滤器(filters):对请求或响应做处理


我们在配置文件中写的断言规则只是字符串,这些字符串会被Predicate Factory读取并处理,转变为路由判断的条件。


例如Path=/user/**是按照路径匹配,这个规则是由org.springframework.cloud.gateway.handler.predicate.PathRoutePredicateFactory类来处理的。


像这样的断言工厂在SpringCloudGateway还有十几个:


image.png

image.png

我们只需要掌握Path这种路由工程就可以了。如果需要用其他的断言规则,可以点击这里进行查看。


4.路由过滤器的配置


GatewayFilter是网关中提供的一种过滤器,可以对进入网关的请求和微服务返回的响应做处理:


a2f3166c90ea463281e9591944c68aaf.png


路由过滤器的种类

Spring提供了31种不同的路由过滤器工厂。例如:


image.png

有其他更多的需求,同样可以点击官方文档。


下面我们以AddRequestHeader 为例来讲解。


需求:给所有进入userservice的请求添加一个请求头:Truth=haiexijun is freaking awesome!


实现方式:在gateway中修改application.yml文件,给userservice的路由添加过滤器:


spring:
  cloud:
    gateway:
      routes:
      - id: user-service 
        uri: lb://userservice 
        predicates: 
        - Path=/user/** 
        filters: # 过滤器
        - AddRequestHeader=Truth, haiexijun is freaking awesome! # 添加请求头


当前过滤器写在userservice路由下,因此仅仅对访问userservice的请求有效。

然后,我们更改一下我们的UserController的代码,来测试一下。


    @GetMapping("/{id}")
    public User queryById(@PathVariable("id") Long id,
                          @RequestHeader(value = "Truth",required = false) String truth) {
        System.out.println(truth);
        return userService.queryById(id);
    }

5285abeb242e4b76ac97ce0531128880.png


默认过滤器

如果要对所有的路由都生效,则可以将过滤器工厂写到default下。格式如下:


spring:
  cloud:
    gateway:
      routes:
      - id: user-service 
        uri: lb://userservice 
        predicates: 
        - Path=/user/**
      default-filters: # 默认过滤项
      - AddRequestHeader=Truth, haiexijun is freaking awesome! 


5. 全局过滤器


上一节学习的过滤器,网关提供了31种,但每一种过滤器的作用都是固定的。如果我们希望拦截请求,做自己的业务逻辑则没办法实现。


全局过滤器作用:


全局过滤器的作用也是处理一切进入网关的请求和微服务响应,与GatewayFilter的作用一样。区别在于GatewayFilter通过配置定义,处理逻辑是固定的;而GlobalFilter的逻辑需要自己写代码实现。


定义方式是实现GlobalFilter接口。


public interface GlobalFilter {
    /**
     *  处理当前请求,有必要的话通过{@link GatewayFilterChain}将请求交给下一个过滤器处理
     *
     * @param exchange 请求上下文,里面可以获取Request、Response等信息
     * @param chain 用来把请求委托给下一个过滤器 
     * @return {@code Mono<Void>} 返回标示当前过滤器业务结束
     */
    Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain);
}


在filter中编写自定义逻辑,可以实现下列功能:


登录状态判断

权限校验

请求限流等

下面来实践一下自定义全局过滤器


需求:定义全局过滤器,拦截请求,判断请求的参数是否满足下面条件:


参数中是否有authorization,


authorization参数值是否为admin


如果同时满足则放行,否则拦截


实现:


在gateway中定义一个过滤器:


package com.haiexijun.gateway.filters;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.annotation.Order;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;
// 过滤器的优先级,越小越先执行
@Order(-1)
// 注册为组件
@Component
public class AuthorizeFilter implements GlobalFilter {
    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        // 1.获取请求参数
        MultiValueMap<String, String> params = exchange.getRequest().getQueryParams();
        // 2.获取authorization参数
        String auth = params.getFirst("authorization");
        // 3.校验
        if ("admin".equals(auth)) {
            // 放行
            return chain.filter(exchange);
        }
        // 4.拦截
        // 4.1.禁止访问,设置状态码
        exchange.getResponse().setStatusCode(HttpStatus.FORBIDDEN);
        // 4.2.结束处理
        return exchange.getResponse().setComplete();
    }
}


2d0a96175e274abebbf102025db413d2.png

e5768d4d11c34cae9196a1bef8b49ece.png


6. 过滤器的执行顺序


请求进入网关会碰到三类过滤器:当前路由的过滤器、DefaultFilter、GlobalFilter


请求路由后,会将当前路由过滤器和DefaultFilter、GlobalFilter,合并到一个过滤器链(集合)中,排序后依次执行每个过滤器:


e46266957baa4f65a51e42396400b302.png


排序的规则是什么呢?


每一个过滤器都必须指定一个int类型的order值,order值越小,优先级越高,执行顺序越靠前。

GlobalFilter通过实现Ordered接口,或者添加@Order注解来指定order值,由我们自己指定

路由过滤器和defaultFilter的order由Spring指定,默认是按照声明顺序从1递增。

当过滤器的order值一样时,会按照 defaultFilter > 路由过滤器 > GlobalFilter的顺序执行。

详细内容,可以查看源码:


org.springframework.cloud.gateway.route.RouteDefinitionRouteLocator#getFilters()方法是先加载defaultFilters,然后再加载某个route的filters,然后合并。


org.springframework.cloud.gateway.handler.FilteringWebHandler#handle()方法会加载全局过滤器,与前面的过滤器合并后根据order排序,组织过滤器链


7. 网关的跨域配置


我们以前在JavaWeb阶段已经学习过跨域问题的解决方案了,那微服务里面为什么要学这个东西呢?


这是因为在微服务当中,所有的请求都要先经过网关,再到微服务。也就是说,跨域请求你不需要在每一个微服务里都去处理,仅仅在网关处理就可以了。但是网关又和我们之前的实现不一样,网关是基于webflux实现的,没有servlet相关的API ,因此我们以前所学的那些解决方案不一定能够适用。


下面先来回顾一下跨域:


跨域:域名不一致就是跨域,主要包括:


域名不同: www.taobao.com 和 www.taobao.org 和 www.jd.com 和 miaosha.jd.com


域名相同,端口不同:localhost:8080和localhost8081


跨域问题:浏览器禁止请求的发起者与服务端发生跨域ajax请求,请求被浏览器拦截的问题


解决方案:CORS,这个以前应该学习过,这里不再赘述了。不知道的小伙伴可以查看https://www.ruanyifeng.com/blog/2016/04/cors.html


解决跨域问题


在gateway服务的application.yml文件中,添加下面的配置:


spring:
  cloud:
    gateway:
      # 。。。
      globalcors: # 全局的跨域处理
        add-to-simple-url-handler-mapping: true # 解决options请求被拦截问题
        corsConfigurations:
          '[/**]':
            allowedOrigins: # 允许哪些网站的跨域请求 
              - "http://localhost:8090"
              - "http://www.leyou.com"
            allowedMethods: # 允许的跨域ajax的请求方式
              - "GET"
              - "POST"
              - "DELETE"
              - "PUT"
              - "OPTIONS"
            allowedHeaders: "*" # 允许在请求中携带的头信息
            allowCredentials: true # 是否允许携带cookie
            maxAge: 360000 # 这次跨域检测的有效期
相关实践学习
基于函数计算快速搭建Hexo博客系统
本场景介绍如何使用阿里云函数计算服务命令行工具快速搭建一个Hexo博客。
相关文章
|
1天前
|
监控 Java 数据库连接
总结Spring Boot面试知识点
Spring Boot是一个基于Spring框架的开源项目,它简化了Spring应用的初始搭建以及开发过程。通过提供“约定优于配置”的方式,Spring Boot可以帮助开发者快速构建出生产级别的Spring应用。
9 0
|
3天前
|
消息中间件 负载均衡 Java
【Spring Cloud 初探幽】
【Spring Cloud 初探幽】
13 1
|
5天前
|
Java 开发者 微服务
Spring Cloud原理详解
【5月更文挑战第4天】Spring Cloud是Spring生态系统中的微服务框架,包含配置管理、服务发现、断路器、API网关等工具,简化分布式系统开发。核心组件如Eureka(服务发现)、Config Server(配置中心)、Ribbon(负载均衡)、Hystrix(断路器)、Zuul(API网关)等。本文讨论了Spring Cloud的基本概念、核心组件、常见问题及解决策略,并提供代码示例,帮助开发者更好地理解和实践微服务架构。此外,还涵盖了服务通信方式、安全性、性能优化、自动化部署、服务网格和无服务器架构的融合等话题,揭示了微服务架构的未来趋势。
29 6
|
9天前
|
JSON Java Apache
Spring Cloud Feign 使用Apache的HTTP Client替换Feign原生httpclient
Spring Cloud Feign 使用Apache的HTTP Client替换Feign原生httpclient
|
9天前
|
负载均衡 Java 开发者
Spring Cloud:一文读懂其原理与架构
Spring Cloud 是一套微服务解决方案,它整合了Netflix公司的多个开源框架,简化了分布式系统开发。Spring Cloud 提供了服务注册与发现、配置中心、消息总线、负载均衡、熔断机制等工具,让开发者可以快速地构建一些常见的微服务架构。
|
11天前
|
消息中间件 Java RocketMQ
Spring Cloud RocketMQ:构建可靠消息驱动的微服务架构
【4月更文挑战第28天】消息队列在微服务架构中扮演着至关重要的角色,能够实现服务之间的解耦、异步通信以及数据分发。Spring Cloud RocketMQ作为Apache RocketMQ的Spring Cloud集成,为微服务架构提供了可靠的消息传输机制。
27 1
|
11天前
|
Dubbo Java 应用服务中间件
Spring Cloud Dubbo: 微服务通信的高效解决方案
【4月更文挑战第28天】在微服务架构的发展中,服务间的高效通信至关重要。Spring Cloud Dubbo 提供了一种基于 RPC 的通信方式,使得服务间的调用就像本地方法调用一样简单。本篇博客将探讨 Spring Cloud Dubbo 的核心概念,并通过具体实例展示其在项目中的实战应用。
14 2
|
11天前
|
监控 Java Sentinel
Spring Cloud Sentinel:概念与实战应用
【4月更文挑战第28天】在分布式微服务架构中,确保系统的稳定性和可靠性至关重要。Spring Cloud Sentinel 为微服务提供流量控制、熔断降级和系统负载保护,有效预防服务雪崩。本篇博客深入探讨 Spring Cloud Sentinel 的核心概念,并通过实际案例展示其在项目中的应用。
22 0
|
11天前
|
Cloud Native Java Nacos
Spring Cloud Nacos:概念与实战应用
【4月更文挑战第28天】Spring Cloud Nacos 是一个基于 Spring Cloud 构建的服务发现和配置管理工具,适用于微服务架构。Nacos 提供了动态服务发现、服务配置、服务元数据及流量管理等功能,帮助开发者构建云原生应用。
18 0