文章目录
- 一、简单需求
- 二、实现过程
- 1、定义接口
- 2、定义实现类
- 1.)web 接口方式初始化类
- 2.)Docker映射文件初始化
- 3.)通过资源文件初始化
- 3、编写初始化逻辑
- 三、单元测试
- 1、单元测试代码
- 2、运行结果
- 3、如何控制优先级
- 四、源码放送
一、简单需求
在CSDN博客自动阅读器-服务端推送技术SSE之简单应用 一文中,我们实现了个人博客文章的后台推送功能。
初始化推送数据是通过接口来实现的,现在我们希望实现如下功能优化:
- 定义多种初始化数据来源,具体而言,有3种方式:①web 接口、②docker映射文件、③本地资源文件
- 支持初始化数据方式的优先级指定。
- 不排除未来会添加其他的初始化方式。例如,通过本地接口提交初始化数据。
二、实现过程
下面我们使用责任链模式来实现上述需求。
1、定义接口
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不可使用异步
,并且需要设置超时时间
,避免长时间无响应的情况下导致的无谓的等待。
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
{@AutowiredWebClient webClient;@Overridepublic 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映射文件初始化
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();@Overridepublic 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.)通过资源文件初始化
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();@Overridepublic 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、编写初始化逻辑
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
{@AutowiredList<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、单元测试代码
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
{@AutowiredList<DataInitor> dataInitors;@BeforeEachpublic void before(){AtomicInteger count = new AtomicInteger();dataInitors.stream().forEach(d -> log.info("{}. {}", count.incrementAndGet(), d));}@Testpublic 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());}@Testpublic 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、运行结果
. ____ _ __ _ _/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \\\/ ___)| |_)| | | | | || (_| | ) ) ) )' |____| .__|_| |_|_| |_\__, | / / / /=========|_|==============|___/=/_/_/_/:: 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-