springboot中责任链模式之简单应用

文章目录

一、简单需求

CSDN博客自动阅读器-服务端推送技术SSE之简单应用 一文中,我们实现了个人博客文章的后台推送功能。

初始化推送数据是通过接口来实现的,现在我们希望实现如下功能优化:

  1. 定义多种初始化数据来源,具体而言,有3种方式:①web 接口、②docker映射文件、③本地资源文件
  2. 支持初始化数据方式的优先级指定。
  3. 不排除未来会添加其他的初始化方式。例如,通过本地接口提交初始化数据。

二、实现过程

下面我们使用责任链模式来实现上述需求。

1、定义接口

java 复制代码
import java.util.List;

import com.fly.demo.entity.Article;
/**
 * 数据初始化
 */
public interface DataInitor
{
    /**
     * 执行初始化
     * 
     * @param articles
     * @return 是否成功
     */
    boolean init(List<Article> articles);
}

2、定义实现类

1.)web 接口方式初始化类

注意: web 接口方式初始化Webclient不可使用异步,并且需要设置超时时间,避免长时间无响应的情况下导致的无谓的等待。

java 复制代码
import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.annotation.Order;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;
import org.springframework.web.reactive.function.client.WebClient;

import com.fly.demo.entity.Article;
import com.fly.demo.entity.BlogData;

import lombok.extern.slf4j.Slf4j;

/**
 * 通过WebApi接口初始化
 */
@Slf4j
@Order(1)
@Component
public class WebApiDataInitor implements DataInitor
{
    @Autowired
    WebClient webClient;
    
    @Override
    public boolean init(List<Article> articles)
    {
        try
        {
            log.info("start init...");
            BlogData blogData = webClient.get()
                .uri("https://00fly.online/upload/data.json")
                .acceptCharset(StandardCharsets.UTF_8)
                .accept(MediaType.APPLICATION_JSON)
                .retrieve()
                .bodyToMono(BlogData.class)
                .timeout(Duration.ofSeconds(10)) // 单独设置超时
                .block();
            if (blogData != null)
            {
                articles.addAll(blogData.getData().getList());
            }
            return !CollectionUtils.isEmpty(articles);
        }
        catch (Exception e)
        {
            log.error(e.getMessage(), e);
            return false;
        }
    }
}

2.)Docker映射文件初始化

java 复制代码
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;

import org.apache.commons.io.IOUtils;
import org.springframework.core.annotation.Order;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;

import com.fly.core.utils.JsonBeanUtils;
import com.fly.demo.entity.Article;
import com.fly.demo.entity.BlogData;

import lombok.extern.slf4j.Slf4j;

/**
 * 通过Docker映射文件初始化
 */
@Slf4j
@Order(2)
@Component
public class DockerDataInitor implements DataInitor
{
    PathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
    
    @Override
    public boolean init(List<Article> articles)
    {
        try
        {
            log.info("start init...");
            Resource[] jsons = resolver.getResources("file:/data/data*.json");
            articles.addAll(Arrays.stream(jsons).map(json -> parseToArticles(json)).flatMap(List::stream).distinct().collect(Collectors.toList()));
            return !CollectionUtils.isEmpty(articles);
        }
        catch (IOException e)
        {
            log.error(e.getMessage(), e);
            return false;
        }
    }
    
    /**
     * 解析Resource为List
     * 
     * @param resource
     * @return
     */
    private List<Article> parseToArticles(Resource resource)
    {
        try (InputStream input = resource.getInputStream())
        {
            String jsonData = IOUtils.toString(input, StandardCharsets.UTF_8);
            return JsonBeanUtils.jsonToBean(jsonData, BlogData.class, true).getData().getList();
        }
        catch (IOException e)
        {
            log.error(e.getMessage(), e);
            return Collections.emptyList();
        }
    }
}

3.)通过资源文件初始化

java 复制代码
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;

import org.apache.commons.io.IOUtils;
import org.springframework.core.annotation.Order;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;

import com.fly.core.utils.JsonBeanUtils;
import com.fly.demo.entity.Article;
import com.fly.demo.entity.BlogData;

import lombok.extern.slf4j.Slf4j;

/**
 * 通过资源文件初始化
 */
@Slf4j
@Order(3)
@Component
public class ResourceDataInitor implements DataInitor
{
    PathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
    
    @Override
    public boolean init(List<Article> articles)
    {
        try
        {
            log.info("start init...");
            Resource[] jsons = resolver.getResources("classpath:*.json");
            articles.addAll(Arrays.stream(jsons).map(json -> parseToArticles(json)).flatMap(List::stream).distinct().collect(Collectors.toList()));
            return !CollectionUtils.isEmpty(articles);
        }
        catch (IOException e)
        {
            log.error(e.getMessage(), e);
            return false;
        }
    }
    
    /**
     * 解析Resource为List
     * 
     * @param resource
     * @return
     */
    private List<Article> parseToArticles(Resource resource)
    {
        try (InputStream input = resource.getInputStream())
        {
            String jsonData = IOUtils.toString(input, StandardCharsets.UTF_8);
            return JsonBeanUtils.jsonToBean(jsonData, BlogData.class, true).getData().getList();
        }
        catch (IOException e)
        {
            log.error(e.getMessage(), e);
            return Collections.emptyList();
        }
    }
}

3、编写初始化逻辑

java 复制代码
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Service;

import com.fly.demo.entity.Article;
import com.fly.demo.service.init.DataInitor;

import lombok.extern.slf4j.Slf4j;

/**
 * DataService
 */
@Slf4j
@Service
public class DataService
{
    @Autowired
    List<DataInitor> dataInitors;
    
    /**
     * 获取url数据列表
     * 
     * @return
     * @throws IOException
     */
    @Cacheable(cacheNames = "data", key = "'articles'", sync = true)
    public List<Article> getArticles()
    {
        AtomicInteger count = new AtomicInteger();
        dataInitors.stream().forEach(d -> log.info("{}. {}", count.incrementAndGet(), d));
        
        // 串行流,有一个DataInitor执行init成功就返回
        List<Article> articles = new ArrayList<>();
        dataInitors.stream()
            .peek(d -> log.info("{}", d.getClass().getName())) // debug
            .anyMatch(d -> d.init(articles));
        log.info("############## articles.size: {} ", articles.size());
        return articles;
    }
}

三、单元测试

为了方便演示,我们编写了单元测试代码

1、单元测试代码

java 复制代码
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;

import com.fly.demo.entity.Article;
import com.fly.demo.service.init.DataInitor;

import lombok.extern.slf4j.Slf4j;

@Slf4j
@SpringBootTest(webEnvironment = WebEnvironment.NONE)
public class DataInitTest
{
    @Autowired
    List<DataInitor> dataInitors;
    
    @BeforeEach
    public void before()
    {
        AtomicInteger count = new AtomicInteger();
        dataInitors.stream().forEach(d -> log.info("{}. {}", count.incrementAndGet(), d));
    }
    
    @Test
    public void testStream()
    {
        // lambda写法, 串行流至少有一个DataInitor执行init成功
        List<Article> articles = new ArrayList<>();
        dataInitors.stream()
            .peek(d -> log.info("{}", d.getClass().getName())) // debug
            .anyMatch(d -> d.init(articles));
        log.info("############## articles.size: {} ", articles.size());
    }
    
    @Test
    public void testCommon()
    {
        // 传统写法
        List<Article> articles = new ArrayList<>();
        for (DataInitor dataInitor : dataInitors)
        {
            log.info("{}", dataInitor.getClass().getName());
            if (dataInitor.init(articles))
            {
                log.info("############## articles.size: {} ", articles.size());
                return;
            }
        }
    }
}

2、运行结果

bash 复制代码
 .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v2.2.4.RELEASE)

2024-12-22 13:04:16.155  INFO 3144 --- [           main] c.f.DataInitTest                         : Starting DataInitTest on 7t9lppye5cj7lud with PID 3144 (started by 00fly in D:\Gitcode\csdn-reader)
2024-12-22 13:04:16.162  INFO 3144 --- [           main] c.f.DataInitTest                         : The following profiles are active: dev
2024-12-22 13:04:18.201  INFO 3144 --- [           main] c.f.c.u.SpringContextUtils               : ###### execute setApplicationContext ######
2024-12-22 13:04:19.535  INFO 3144 --- [           main] o.s.s.c.ThreadPoolTaskScheduler          : Initializing ExecutorService 'taskScheduler'
2024-12-22 13:04:19.663  INFO 3144 --- [           main] c.f.DataInitTest                         : Started DataInitTest in 4.551 seconds (JVM running for 10.165)
2024-12-22 13:04:20.366  INFO 3144 --- [           main] c.f.DataInitTest                         : 1. com.fly.demo.service.init.WebApiDataInitor@2881ad47
2024-12-22 13:04:20.367  INFO 3144 --- [           main] c.f.DataInitTest                         : 2. com.fly.demo.service.init.DockerDataInitor@37fdfb05
2024-12-22 13:04:20.367  INFO 3144 --- [           main] c.f.DataInitTest                         : 3. com.fly.demo.service.init.ResourceDataInitor@5e39850
2024-12-22 13:04:20.374  INFO 3144 --- [           main] c.f.DataInitTest                         : com.fly.demo.service.init.WebApiDataInitor
2024-12-22 13:04:20.374  INFO 3144 --- [           main] c.f.d.s.i.WebApiDataInitor               : start init...
2024-12-22 13:04:24.000  INFO 3144 --- [           main] c.f.DataInitTest                         : ############## articles.size: 126 
2024-12-22 13:04:26.082  INFO 3144 --- [extShutdownHook] o.s.s.c.ThreadPoolTaskScheduler          : Shutting down ExecutorService 'taskScheduler'

3、如何控制优先级

细心的同鞋,已经发现了在我们的实现类中使用了@Order 注解,仔细关心上面的日志输出,我们发现order的取值会影响 List<DataInitor> dataInitors的实现类的排列顺序,假如我们需要把Docker映射文件初始化优先级提升,只需要把order改小,改为0或-1均可,大家可以动手尝试!

四、源码放送

https://gitcode.com/00fly/csdn-reader


有任何问题和建议,都可以向我提问讨论,大家一起进步,谢谢!

-over-

相关推荐
苹果酱056735 分钟前
Golang的文件解压技术研究与应用案例
java·vue.js·spring boot·mysql·课程设计
Q_19284999062 小时前
基于Spring Boot的个人健康管理系统
java·spring boot·后端
会说法语的猪4 小时前
springboot实现图片上传、下载功能
java·spring boot·后端
m0_748239834 小时前
基于web的音乐网站(Java+SpringBoot+Mysql)
java·前端·spring boot
m0_748234084 小时前
Spring Boot教程之三十一:入门 Web
前端·spring boot·后端
昙鱼5 小时前
springboot创建web项目
java·前端·spring boot·后端·spring·maven
天之涯上上5 小时前
JAVA开发 在 Spring Boot 中集成 Swagger
java·开发语言·spring boot
白宇横流学长5 小时前
基于SpringBoot的停车场管理系统设计与实现【源码+文档+部署讲解】
java·spring boot·后端
kirito学长-Java5 小时前
springboot/ssm太原学院商铺管理系统Java代码编写web在线购物商城
java·spring boot·后端
中草药z6 小时前
【Spring】深入解析 Spring 原理:Bean 的多方面剖析(源码阅读)
java·数据库·spring boot·spring·bean·源码阅读