【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

相关文章

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Java easyExcel实现导入多sheet的Excel

《JavaeasyExcel实现导入多sheet的Excel》这篇文章主要为大家详细介绍了如何使用JavaeasyExcel实现导入多sheet的Excel,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录1.官网2.Excel样式3.代码1.官网easyExcel官网2.Excel样式3.代码

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文