【quarkus系列】实战自定义注解实现策略模式分发

2024-06-06 07:20

本文主要是介绍【quarkus系列】实战自定义注解实现策略模式分发,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 序言
  • 自定义注解
  • 业务接口
  • 渠道消息实现
  • 策略分发
  • 测试
  • 知识扩展
    • @Any
    • AnnotationLiteral 应用场景和语法

序言

策略模式大家都应该了解或者使用过,此篇文章中就不再阐述,之前springboot项目中小编也真正的实战应用过。现在换Quarkus框架开发项目,为了后期的项目业务的可扩展以及高可用,就构思并了解如何在Quarkus中使用,软件设计思想最重要,换组件无非换种语法而已,但是也需要摸索,排坑之类,下面给出一个简单的示例demo:模拟向不同的平台类型发送消息

自定义注解

基于Quarkus自定义注解的语法格式,与java语法基本一致,毕竟遵循 MicroProfile 和 Jakarta EE 的规范。

  • 核心代码
@Qualifier
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.FIELD, ElementType.PARAMETER, ElementType.METHOD})
public @interface Platform {PlatformTypeEnum type();
}
  • 平台类型枚举
@AllArgsConstructor
@Getter
public enum PlatformTypeEnum {//企微WEWORK(1, "企微"),//小红书RED_BOOK(2, "小红书"),;private final Integer type;private final String desc;
}

业务接口

抽象消息业务接口,应用于多平台渠道实现

public interface MessageService {String sendMsg(String msg);}

渠道消息实现

  • 企业微信渠道实现
@Platform(type = PlatformTypeEnum.WEWORK)
@Dependent
public class WeWorkMessage implements MessageService {@InjectMessageRepo messageRepo;@Overridepublic String sendMsg(String msg) {//TODOreturn messageRepo.getMessage() + ", weWork";}
}
  • 小红书渠道消息实现
@Platform(type = PlatformTypeEnum.RED_BOOK)
@Dependent
public class RedBookMessage implements MessageService {@Overridepublic String sendMsg(String msg) {//TODOreturn "redBook";}
}

策略分发

  • 方式一:switch-case
  public MessageService getStrategy(PlatformTypeEnum platform) {switch (platform) {case WEWORK:return messageServices.select(WeWorkMessage.class).get();case RED_BOOK:return messageServices.select(RedBookMessage.class).get();default:return null;}}

这种方式违反了开闭原则,每次增加子类都需要修改此处的代码;

  • 方式二:动态分发
@ApplicationScoped
public class MessageStrategy {@Inject@AnyInstance<MessageService> messageServices;public MessageService getStrategy(PlatformTypeEnum platform) {return messageServices.select(new PlatformLiteral(platform)).get();}private static class PlatformLiteral extends AnnotationLiteral<Platform> implements Platform {private final PlatformTypeEnum platform;public PlatformLiteral(PlatformTypeEnum platform) {this.platform = platform;}@Overridepublic PlatformTypeEnum type() {return platform;}}
}

getStrategy 方法使用 messageServices.select 方法来动态选择符合条件的 MessageService 实例。
PlatformLiteral 用于在运行时创建一个带有具体值的 @Platform 注解。
PlatformLiteral 类扩展了 AnnotationLiteral 并实现了 Platform 接口。它接受一个 PlatformTypeEnum 值并重写 type 方法返回该值。通过这种方式,PlatformLiteral 在运行时创建了一个带有特定 PlatformTypeEnum 值的 @Platform 注解实例。

  • 优点
    这个机制允许在运行时动态选择不同的 MessageService 实现,而无需在编译时确定具体的实现类。

测试

@Path("/message")
@Tag(name = "消息")
public class MessageController {@InjectMessageStrategy messageStrategy;@Path("/get-msg")@GETpublic Response<String> getMsg(@QueryParam("platform") PlatformTypeEnum platform) {String test = messageStrategy.getStrategy(platform).sendMsg("test");return Response.ok(test);}
}

可以看到messageService中存在两个实现类
在这里插入图片描述
使用swagger测试,分布使用企微、小红书入参得到的结果如图:
在这里插入图片描述

  • 小红书,消息渠道分发
    在这里插入图片描述

知识扩展

基于方式二实现扩展使用的注解语法以及相关原理

@Any

  • 应用场景
    @Any 注解在 CDI(Context and Dependency Injection)中用于注入所有符合特定类型的 Bean 实例,而不考虑其他限定符。它通常用于动态选择或处理多个实现。

  • 语法

@Inject
@Any
Instance<SomeType> someInstances;
  • 原理
    @Any 告诉 CDI 容器注入所有符合类型 SomeType 的 Bean 实例,即使这些实例有其他限定符。

  • @Any注解的源码,是不是很熟悉呀

@Qualifier
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.METHOD, ElementType.FIELD, ElementType.PARAMETER})
@Documented
public @interface Any {public static final class Literal extends AnnotationLiteral<Any> implements Any {public static final Literal INSTANCE = new Literal();private static final long serialVersionUID = 1L;public Literal() {}}
}

AnnotationLiteral 应用场景和语法

  • 应用场景
    AnnotationLiteral 用于在运行时动态创建注解实例。它通常用于 CDI 的动态选择,特别是当需要根据条件在运行时选择特定 Bean 实例时。

  • 语法

public class SomeAnnotationLiteral extends AnnotationLiteral<SomeAnnotation> implements SomeAnnotation {private final SomeType value;public SomeAnnotationLiteral(SomeType value) {this.value = value;}@Overridepublic SomeType value() {return value;}
}
  • 原理
    AnnotationLiteral 是一个抽象类,用于表示注解的具体实例,提供了注解的具体实现,使得可以在运行时生成注解实例,用于 CDI 的选择操作。原理是利用泛型和反射机制,它通过创建一个匿名内部类来实现对注解实例的动态创建,通过扩展 AnnotationLiteral 类并实现注解接口的方法,可以动态创建带有具体值的注解实例,而不是依赖于编译时注解处理器生成的实例。

这篇关于【quarkus系列】实战自定义注解实现策略模式分发的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1035448

相关文章

Python中edge-tts实现便捷语音合成

《Python中edge-tts实现便捷语音合成》edge-tts是一个功能强大的Python库,支持多种语言和声音选项,本文主要介绍了Python中edge-tts实现便捷语音合成,具有一定的参考价... 目录安装与环境设置文本转语音查找音色更改语音参数生成音频与字幕总结edge-tts 是一个功能强大的

Java实现按字节长度截取字符串

《Java实现按字节长度截取字符串》在Java中,由于字符串可能包含多字节字符,直接按字节长度截取可能会导致乱码或截取不准确的问题,下面我们就来看看几种按字节长度截取字符串的方法吧... 目录方法一:使用String的getBytes方法方法二:指定字符编码处理方法三:更精确的字符编码处理使用示例注意事项方

使用Python和PaddleOCR实现图文识别的代码和步骤

《使用Python和PaddleOCR实现图文识别的代码和步骤》在当今数字化时代,图文识别技术的应用越来越广泛,如文档数字化、信息提取等,PaddleOCR是百度开源的一款强大的OCR工具包,它集成了... 目录一、引言二、环境准备2.1 安装 python2.2 安装 PaddlePaddle2.3 安装

嵌入式Linux之使用设备树驱动GPIO的实现方式

《嵌入式Linux之使用设备树驱动GPIO的实现方式》:本文主要介绍嵌入式Linux之使用设备树驱动GPIO的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、设备树配置1.1 添加 pinctrl 节点1.2 添加 LED 设备节点二、编写驱动程序2.1

Android 实现一个隐私弹窗功能

《Android实现一个隐私弹窗功能》:本文主要介绍Android实现一个隐私弹窗功能,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 效果图如下:1. 设置同意、退出、点击用户协议、点击隐私协议的函数参数2. 《用户协议》、《隐私政策》设置成可点击的,且颜色要区分出来res/l

spring IOC的理解之原理和实现过程

《springIOC的理解之原理和实现过程》:本文主要介绍springIOC的理解之原理和实现过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、IoC 核心概念二、核心原理1. 容器架构2. 核心组件3. 工作流程三、关键实现机制1. Bean生命周期2.

Redis实现分布式锁全解析之从原理到实践过程

《Redis实现分布式锁全解析之从原理到实践过程》:本文主要介绍Redis实现分布式锁全解析之从原理到实践过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、背景介绍二、解决方案(一)使用 SETNX 命令(二)设置锁的过期时间(三)解决锁的误删问题(四)Re

一文带你搞懂Redis Stream的6种消息处理模式

《一文带你搞懂RedisStream的6种消息处理模式》Redis5.0版本引入的Stream数据类型,为Redis生态带来了强大而灵活的消息队列功能,本文将为大家详细介绍RedisStream的6... 目录1. 简单消费模式(Simple Consumption)基本概念核心命令实现示例使用场景优缺点2

Java根据IP地址实现归属地获取

《Java根据IP地址实现归属地获取》Ip2region是一个离线IP地址定位库和IP定位数据管理框架,这篇文章主要为大家详细介绍了Java如何使用Ip2region实现根据IP地址获取归属地,感兴趣... 目录一、使用Ip2region离线获取1、Ip2region简介2、导包3、下编程载xdb文件4、J

PyQt5+Python-docx实现一键生成测试报告

《PyQt5+Python-docx实现一键生成测试报告》作为一名测试工程师,你是否经历过手动填写测试报告的痛苦,本文将用Python的PyQt5和python-docx库,打造一款测试报告一键生成工... 目录引言工具功能亮点工具设计思路1. 界面设计:PyQt5实现数据输入2. 文档生成:python-