SpringBoot+SpringCloud+nacos+gateway+mybatis搭建一个微服务(实习生的日常)

发布于:2022-11-09 ⋅ 阅读:(9) ⋅ 点赞:(0) ⋅ 评论:(0)

SpringBoot+SpringCloud+nacos+gateway+mybatis搭建一个微服务

最近公司进行考核,其中就有搭建微服务并完成一些简单的业务,之前都是从git上拉一个,让我自己搭建还真是走了不少弯路。最后搭建起来回头看看也没有那么难

Nacos说明

Nacos除了可以做注册中心,同样可以做配置管理来使用。
nacos需要安装,不理解nacos可以去官方了解和下载
nacos官网

Nacos是一个快速实现动态服务发现、服务配置、服务元数据及流量管理。

Getawaty说明

它其实就充当一个网关,可以将用户请求过滤以及修改然后再通过Getaway去转发到对应的请求服务上。(它本身也是一个服务,也要注册到nacos)
Gateway网关是我们服务的守门神,所有微服务的统一入口。

网关的核心功能特性

  • 请求路由
  • 权限控制
  • 限流

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

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

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

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

  • gateway
  • zuul

Zuul是基于Servlet的实现,属于阻塞式编程。而SpringCloudGateway则是基于Spring5中提供的WebFlux,属于响应式编程的实现,具备更好的性能。

Feign远程调用说明

Feign是一个声明式的http客户端,官方地址:feign官网
作用就是帮助我们优雅的实现http请求的发送

开始搭建项目

SpringCloud微服务,虽然很多服务,但是服务之间使用的开发环境是相同的,所以Maven依赖也是有很多共同的。那么可以建一个父 Maven项目,然后配置依赖。在大项目中创建各个微服务项目。

其实就是一个父模块和若干子模块(maven模块)
在这里插入图片描述

父模块

pom内容

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.3.9.RELEASE</version>
        <relativePath/>
    </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>Hoxton.SR10</spring-cloud.version>
        <mysql.version>5.1.47</mysql.version>
        <mybatis.version>2.1.1</mybatis.version>
    </properties>

    <dependencyManagement>
        <dependencies>
            <!-- springCloud -->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring-cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
            <!--nacos的管理依赖-->
            <dependency>
                <groupId>com.alibaba.cloud</groupId>
                <artifactId>spring-cloud-alibaba-dependencies</artifactId>
                <version>2.2.5.RELEASE</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
            <!-- mysql驱动 -->
            <dependency>
                <groupId>mysql</groupId>
                <artifactId>mysql-connector-java</artifactId>
                <version>${mysql.version}</version>
            </dependency>
            <!--mybatis-->
            <dependency>
                <groupId>org.mybatis.spring.boot</groupId>
                <artifactId>mybatis-spring-boot-starter</artifactId>
                <version>${mybatis.version}</version>
            </dependency>
        </dependencies>
    </dependencyManagement>
    <dependencies>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>
    </dependencies>

创建服务user,并配置依赖

创建 application.properties并配置

server:
  port: 8081
spring:
  datasource:
    url: jdbc:mysql://localhost:3306/cloud_user?useSSL=false
    username: root
    password: 123456
    driver-class-name: com.mysql.jdbc.Driver
  application:
    name: userservice
  cloud:
    nacos:
      server-addr: localhost:8848 # nacos服务地址
#      discovery:
#        namespace: 4d6ce343-9e1b-44df-a90f-2cf2b6b3d177 # dev环境
#        ephemeral: false # 是否是临时实例
mybatis:
  type-aliases-package: com.it.product.pojo
  configuration:
    map-underscore-to-camel-case: true
logging:
  level:
    com.it: debug
  pattern:
    dateformat: MM-dd HH:mm:ss:SSS
#eureka:
#  client:
#    service-url:  # eureka的地址信息
#      defaultZone: http://127.0.0.1:10086/eureka
userservice:
  ribbon:
    NFLoadBalancerRuleClassName: com.alibaba.cloud.nacos.ribbon.NacosRule  # 负载均衡规则
ribbon:
  eager-load:
    enabled: true # 开启饥饿加载
    clients: # 指定饥饿加载的服务名称
      - productservice
feign:
  httpclient:
    enabled: true # 支持HttpClient的开关
    max-connections: 200 # 最大连接数
    max-connections-per-route: 50 # 单个路径的最大连接数
    #main:
    #  allow-bean-definition-overriding: true # 一个服务只能用@FeignClient使用一次。现在可以使用多个

配置pom.xml

 <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>
        <!--mybatis-->
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
        </dependency>

        <!-- nacos客户端依赖包 -->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>

        <!--feign客户端依赖-->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
        </dependency>
        <!--引入HttpClient依赖-->
        <dependency>
            <groupId>io.github.openfeign</groupId>
            <artifactId>feign-httpclient</artifactId>
        </dependency>
        <!--引入feign的统一api-->
        <dependency>
            <groupId>com.it</groupId>
            <artifactId>feign-api</artifactId>
        </dependency>
        <dependency>
            <groupId>com.it</groupId>
            <artifactId>feign-api</artifactId>
            <version>1.0-SNAPSHOT</version>
            <scope>compile</scope>
        </dependency>
    </dependencies>
    <build>
        <finalName>app</finalName>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

然后开始写业务代码…结构如图
在这里插入图片描述
然后启动user-server,看一看nacos注册上没有
nacos地址:localhost:8848 ,初始账号密码都是nacos。

创建服务product,并配置依赖

同上我们创建product,这里就不在写了。注意端口号改一下**
这里配置都是一样的,只是user-service多了feign的依赖和feign.httpclient配置(因为我是用user-service去调用product服务的)

服务搭建完毕之后开始搭建feign

注意需要将feign抽取成一个独立的模块(松耦合)
在这里插入图片描述
userservice调用productservice的feign

/**
 * 调用productservice的feign
 */
@FeignClient(value = "productservice")
public interface ProductClient {
    /**
     * 根据id查询角色
     * @param roleId
     * @return
     */
    @GetMapping("/role/{roleId}")
    Role findByRoleId(@PathVariable("roleId") Long roleId);

    /**
     * 根据id查询部门
     * @param deptId
     * @return
     */
    @GetMapping("/dept/{deptId}")
    Dept findByDeptId(@PathVariable("deptId") Long deptId);
}

注意这里pojo还要有被调用服务的实体类

pom文件

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

注意user-service去远程调用productservice,要在user-service启动类加上
@EnableFeignClients(clients = ProductClient.class,defaultConfiguration = DefaultFeignConfiguration.class)
注解

接下来 配置网关

也是作为一个独立的模块(需要启动类)
application.yml

server:
  port: 10010
logging:
  level:
    cn.itcast: debug
  pattern:
    dateformat: MM-dd HH:mm:ss:SSS
spring:
  application:
    name: gateway
  cloud:
    nacos:
      server-addr: localhost:8848 # nacos地址
    gateway:
      routes:
        - id: user-service # 路由标示,必须唯一
          uri: lb://userservice # 路由的目标地址
          predicates: # 路由断言,判断请求是否符合规则
            - Path=/user/** # 路径断言,判断路径是否是以/user开头,如果是则符合
        - id: product-service
          uri: lb://productservice
          predicates:
            - Path=/role/**
      default-filters:
        - AddRequestHeader=Truth,wwwwwwwwwwwwww! #添加请求头示例

pom文件 配置nacos服务注册依赖和网关依赖

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

然后可以加一个全局过滤器(可以用于校验)


/**
 * 自定义全局过滤器
 */
 //@Order(-1)  //优先级 数值越小 优先级越低
@Component
public class AuthorizeFilter implements GlobalFilter, Ordered {
    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        // 1.获取请求参数
        ServerHttpRequest request = exchange.getRequest();
        MultiValueMap<String, String> params = request.getQueryParams();
        // 2.获取参数中的 authorization 参数
        String auth = params.getFirst("authorization");
        // 3.判断参数值是否等于 admin
        if ("admin".equals(auth)) {
            // 4.是,放行
            return chain.filter(exchange);
        }
        // 5.否,拦截
        // 5.1.设置状态码
        exchange.getResponse().setStatusCode(HttpStatus.UNAUTHORIZED);
        // 5.2.拦截请求
        return exchange.getResponse().setComplete();
    }

    @Override
    public int getOrder() {
        return -1;
    }
}

其实上述代码就是,在请求的url后面需要加上authorization=admin才能放行
例如 localhost:10010/user/1?authorization=admin

到此,SpringBoot+SpringCloud+nacos+gateway+mybatis微服务就搭建完成了,开冲!