老男孩的成长之路Java架构技术进阶

Spring Cloud Alibaba微服务生态的基础实践

2020-09-24  本文已影响0人  Java入门到入坟

推荐阅读:

一、背景

二、初识Spring Cloud Alibaba

  1. 2016 年,阿里全面拥抱 Spring Boot;
  2. 2017 年 12 月,Spring Cloud Alibaba 立项并顺利进入 Spring Cloud 孵化器。
  3. 2019 月 10 月 3 日,Spring Cloud Alibaba 正式 "挂牌" Spring 官方。
同 Spring Cloud 一样,Spring Cloud Alibaba 也是一套微服务解决方案,包含开发分布式应用微服务的必需组件,方便开发者通过 Spring Cloud 编程模型轻松使用这些组件来开发分布式应用服务。 (图片来源:阿里云云栖号) 图片来源阿里云云栖号​

三、Nacos的基础实践

Nacos 支持几乎所有主流类型的“服务”的发现、配置和管理:

  1. Kubernetes Serviceg
  2. RPC & Dubbo RPC Service
  3. Spring Cloud RESTful Service Nacos生态图[(来自Nacos官网)] 来自Nacos官网

3.1 安装Nacos并启动服务

# 下载nacos最新版
 wget https://github.com/alibaba/nacos/releases/download/1.3.2/nacos-server-1.3.2.tar.gz
# 解压文件
 tar -xvf  nacos-server-1.3.2.tar.gz
# 通过nacos-mysql.sql脚本建立数据库
 cd nacos/conf
 vim nacos-mysql.sql
 /******************************************/
/*   数据库全名 = nacos_config   */
/*   表名称 = config_info   */
/******************************************/
CREATE TABLE `config_info` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'id',
  `data_id` varchar(255) NOT NULL COMMENT 'data_id',
  `group_id` varchar(255) DEFAULT NULL,
  `content` longtext NOT NULL COMMENT 'content',
  `md5` varchar(32) DEFAULT NULL COMMENT 'md5',
  `gmt_create` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
  `gmt_modified` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间',
  `src_user` text COMMENT 'source user',
  `src_ip` varchar(20) DEFAULT NULL COMMENT 'source ip',
  `app_name` varchar(128) DEFAULT NULL,
  `tenant_id` varchar(128) DEFAULT '' COMMENT '租户字段',
  `c_desc` varchar(256) DEFAULT NULL,
  `c_use` varchar(64) DEFAULT NULL,
  `effect` varchar(64) DEFAULT NULL,
  `type` varchar(64) DEFAULT NULL,
  `c_schema` text,
  PRIMARY KEY (`id`),
...

 cd nacos/conf
 # 编辑配置文件
 vim application.properties

#*************** Spring Boot Related Configurations ***************#
### Default web context path:
server.servlet.contextPath=/nacos
### Default web server port:
server.port=8848

#*************** Network Related Configurations ***************#
### If prefer hostname over ip for Nacos server addresses in cluster.conf:
# nacos.inetutils.prefer-hostname-over-ip=false

### Specify local server's IP:
# nacos.inetutils.ip-address=

#*************** Config Module Related Configurations ***************#
### If use MySQL as datasource:
 spring.datasource.platform=mysql

### Count of DB:
 db.num=1

### Connect URL of DB:
 db.url.0=jdbc:mysql://127.0.0.1:3306/nacos_config?characterEncoding=utf8&connectTimeout=1000&socketTimeout=3000&autoReconnect=true&useUnicode=true&useSSL=false&serverTimezone=UTC
 db.user=root
 db.password=root
cd nacos/bin
# 单机运行
sh startup.sh -m standalone

3.2 建立微服务并向Nacos注册服务

<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.9.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>nacos.democonsumer</groupId>
    <artifactId>sentinel</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>sentinel</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
        <spring-cloud-alibaba.version>2.2.1.RELEASE</spring-cloud-alibaba.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>

    <dependencyManagement>
        <dependencies>
            <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>
        </dependencies>
    </dependencyManagement>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>
# 端口
server:
  port: 8083

spring:
  application:
    name: goods-service
  cloud:
    # nacos服务注册
    nacos:
      discovery:
        server-addr: 172.16.109.118:8848
// 通过 Spring Cloud 原生注解 @EnableDiscoveryClient 开启服务注册发现功能
@EnableDiscoveryClient
@SpringBootApplication
public class DemoServiceApplication {

    public static void main(String[] args) {

        SpringApplication.run(DemoServiceApplication.class, args);
    }

}
/**
 * 商品信息微服务-模拟返回商品列表
 */
@RestController
@RequestMapping("api/goods")
public class GoodsService {
    public static final Logger logger = LoggerFactory.getLogger(GoodsService.class);
    // 返回商品列表
    @GetMapping
    public List<Goods> getAllGoods(HttpServletRequest httpServletRequest) {
        List<Goods> goods = new ArrayList<>();
        goods.add(new Goods("电脑", 10));
        goods.add(new Goods("手机", 20));
        goods.add(new Goods("书籍", 30));
        logger.info("服务被调用:"+httpServletRequest.getRequestURI());
        return goods;
    }

}

3.3 建立微服务消费者进行服务调用

...
 <!-- 加入Nacos Discovery Client 依赖 -->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>
...
# 端口
server:
  port: 8090
# Spring配置
spring:
  application:
    name: user-consumer
  cloud:
    nacos:
      discovery:
        server-addr: 172.16.109.118:8848
        enabled: true

# 调用微服务超时时间设置
ribbon:
  ConnectTimeout: 5000
  ReadTimeout: 5000

# feign日志以什么级别监控哪个接口
logging:
  level:
    nacos.democonsumer.GoodService : debug
# 商品微服务地址
service:
  url=http://goods-service/
@EnableDiscoveryClient
@SpringBootApplication
public class DemoConsumerApplication {

    // 向Spring注册一个RestTemplate Bean
    @Bean
    // 负载均衡
    @LoadBalanced
    RestTemplate restTemplate(){
        return new RestTemplate();
    }

    public static void main(String[] args) {
        SpringApplication.run(DemoConsumerApplication.class, args);
    }
}
/**
 * 用户消费者--调用nacos服务中心的商品信息微服务,并对外提供RestFul接口测试
 */
@RestController
@RequestMapping("user/")
public class UserConsumer {

    @Autowired
    private RestTemplate restTemplate;

    @Value("${service.url}")
    private String url;

    @GetMapping("/goods")
    public User getUserGoods() {
        User user = new User();
        // 调用商品微服务
        Object response = restTemplate.getForEntity(url + "api/goods", Object.class).getBody();
        try {
            user.setName("jack");
            user.setGoods((List<GoodsDTO>) response);
        } catch (Exception e){
            throw new RuntimeException(e.getMessage());
        }
        return user;
    }
}

3.4 Nacos小结

四、Sentinel的基础实践

4.1 安装Sentinel监控

 # 下载服务端
 wget https://github.com/alibaba/Sentinel/releases/download/v1.8.0/sentinel- dashboard-1.8.0.jar
 # 启动服务(默认端口为8080)
 java -jar sentinel-dashboard-1.8.0.jar

4.2 通过Sentinel对微服务提供方进行流量控制

<project>
        ...
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-sentinel</artifactId>
        </dependency>
        ...
</project>

server:
  port: 8083

spring:
  application:
    name: goods-service
  cloud:
    # nacos服务注册
    nacos:
      discovery:
        server-addr: 172.16.109.118:8848
    # sentinel服务
    sentinel:
      transport:
        dashboard: 172.16.109.118:8080

4.2.1 通过JMeter模拟高并发流量

4.2.2 设置流控规则进行流量控制

4.2.3 设置降级规则进行流量控制

慢调用比例 (SLOW_REQUEST_RATIO):选择以慢调用比例作为阈值,需要设置允许的慢调用 RT(即最大的响应时间),请求的响应时间大于该值则统计为慢调用。当单位统计时长(statIntervalMs)内请求数目大于设置的最小请求数目,并且慢调用的比例大于阈值,则接下来的熔断时长内请求会自动被熔断。经过熔断时长后熔断器会进入探测恢复状态(HALF-OPEN 状态),若接下来的一个请求响应时间小于设置的慢调用 RT 则结束熔断,若大于设置的慢调用 RT 则会再次被熔断。

4.3 通过Sentinel对微服务调用方进行流量控制

<-- pom.xml -->
<!-- https://mvnrepository.com/artifact/org.springframework.cloud/spring-cloud-starter-openfeign -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
            <version>2.2.5.RELEASE</version>
        </dependency>
server:
  port: 8090

spring:
  application:
    name: user-consumer
  cloud:
    nacos:
      discovery:
        server-addr: 172.16.109.118:8848
        # sentinel服务
    sentinel:
      transport:
        dashboard: 172.16.109.118:8080

# 调用微服务超时时间设置
ribbon:
  ConnectTimeout: 5000
  ReadTimeout: 5000

# feign日志以什么级别监控哪个接口
logging:
  level:
    nacos.democonsumer.GoodService : debug

# 启用sentinel
feign:
  sentinel:
    enabled: true

/**
 * 主启动类
 */
 // 启用Feign组件
@EnableFeignClients
@EnableDiscoveryClient
@SpringBootApplication
public class DemoConsumerApplication {

    public static void main(String[] args) {
        SpringApplication.run(DemoConsumerApplication.class, args);
    }
}

/**
 * 映射商品微服务接口
 */
@Component
// 增加服务容错处理,指定服务熔断时处理的类名
@FeignClient(value="goods-service",fallback =FallBackService.class )
public interface GoodService {

    @GetMapping(value="/api/goods")
    List<GoodsDTO> getGoods();

}

/**
 * 服务熔断处理:返回空值
 */
@Component
public class FallBackService implements GoodService {
    private final static Logger logger= LoggerFactory.getLogger(FallBackService.class) ;

    @Override
    public List<GoodsDTO> getGoods() {
        logger.info("服务已熔断...");
        return new ArrayList<>();
    }
}

/**
 * 用户消费者--调用nacos服务中心的商品服务,并对外提供RestFul接口测试
 */
@RestController
@RequestMapping("user/")
public class UserConsumer {
    // 注入商品微服务接品
    @Autowired
    private GoodService goodService;

    @GetMapping("/goods")
    public User getUserGoods() {
        User user = new User();
        // 通过GoodsService接口调用商品微服务
        try {
            List<GoodsDTO> goods = goodService.getGoods();
            user.setName("jack");
            user.setGoods(goods);
        } catch (Exception e){
            throw new RuntimeException(e.getMessage());
        }
        return user;
    }

}

4.3.1 对服务调用者中的服务提供方进行流量控制

4.4 Sentinel实践小结

上一篇下一篇

猜你喜欢

热点阅读