基于spring如何實現(xiàn)事件驅動實例代碼
干貨點
通過閱讀該篇博客,你可以了解了解java的反射機制、可以了解如何基于spring生命周期使用自定義注解解決日常研發(fā)問題。具體源碼可以點擊鏈接。
問題描述
在日常研發(fā)中,經(jīng)常會遇見業(yè)務A的某個action被觸發(fā)后,同時觸發(fā)業(yè)務B的action的行為,這種單對單的形式可以直接在業(yè)務A的action執(zhí)行結束后直接調(diào)用業(yè)務B的action,那么如果是單對多的情況呢?
方案解決
這里提供一種在日常研發(fā)中經(jīng)常使用到的機制,基于spring實現(xiàn)的事件驅動,即在業(yè)務A的action執(zhí)行完,拋出一個事件,而業(yè)務B、C、D等監(jiān)聽到該事件后處理相應的業(yè)務。
場景范例
這里提供一個場景范例,該范例基于springboot空殼項目實現(xiàn),具體可以查看源碼,此處只梳理關鍵步驟。
步驟一:
定義一個注解,標志接收事件的注解,即所有使用了該注解的函數(shù)都會在對應事件被拋出的時候被調(diào)用,該注解實現(xiàn)比較簡單,代碼如下
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc 接收事件的注解
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface ReceiveAnno {
// 監(jiān)聽的事件
Class clz();
}
如果想了解注解多個參數(shù)的意義是什么的可以點擊鏈接查看博主之前寫過文章。
定義事件接口
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc
*/
public interface IEvent {
}
所有事件都需要實現(xiàn)該接口,主要是為了后面泛型和類型識別。
定義MethodInfo
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc
*/
public class MethodInfo {
public Object obj;
public Method method;
public static MethodInfo valueOf(Method method, Object obj) {
MethodInfo info = new MethodInfo();
info.method = method;
info.obj = obj;
return info;
}
public Object getObj() {
return obj;
}
public Method getMethod() {
return method;
}
}
該類只是做了Object和Method的封裝,沒有其他作用。
步驟二:
實現(xiàn)一個事件容器,該容器的作用是存放各個事件以及需要觸發(fā)的各個業(yè)務的method的對應關系。
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc 事件容器
*/
public class EventContainer {
private static Map<Class<IEvent>, List<MethodInfo>> eventListMap = new HashMap<>();
public static void addEventToMap(Class clz, Method method, Object obj) {
List<MethodInfo> methodInfos = eventListMap.get(clz);
if (methodInfos == null) {
methodInfos = new ArrayList<>();
eventListMap.put(clz, methodInfos);
}
methodInfos.add(MethodInfo.valueOf(method, obj));
}
public static void submit(Class clz) {
List<MethodInfo> methodInfos = eventListMap.get(clz);
if (methodInfos == null) {
return;
}
for (MethodInfo methodInfo : methodInfos) {
Method method = methodInfo.getMethod();
try {
method.setAccessible(true);
method.invoke(methodInfo.getObj());
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
}
其中的addEventToMap函數(shù)的作用是將對應的事件、事件觸發(fā)后需要觸發(fā)的對應業(yè)務內(nèi)的Method存放在eventListMap內(nèi);而submit函數(shù)會在其他業(yè)務類內(nèi)拋出事件的時候被調(diào)用,而作用是從eventListMap中取出對應的Method,并通過反射觸發(fā)。
步驟三:
實現(xiàn)事件處理器,該事件處理器的作用是在bean被spring容器實例化后去判斷對應的bean是否有相應函數(shù)加了@ReceiveAnno注解,如果有則從中取出對應的Event并放入EventContainer中。
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc 事件處理器
*/
@Component
public class EventProcessor extends InstantiationAwareBeanPostProcessorAdapter {
@Override
public boolean postProcessAfterInstantiation(Object bean, String beanName) throws BeansException {
ReflectionUtils.doWithLocalMethods(bean.getClass(), new ReflectionUtils.MethodCallback() {
@Override
public void doWith(Method method) throws IllegalArgumentException, IllegalAccessException {
ReceiveAnno anno = method.getAnnotation(ReceiveAnno.class);
if (anno == null) {
return;
}
Class clz = anno.clz();
try {
if (!IEvent.class.isInstance(clz.newInstance())) {
FormattingTuple message = MessageFormatter.format("{}沒有實現(xiàn)IEvent接口", clz);
throw new RuntimeException(message.getMessage());
}
} catch (InstantiationException e) {
e.printStackTrace();
}
EventContainer.addEventToMap(clz, method, bean);
}
});
return super.postProcessAfterInstantiation(bean, beanName);
}
}
關于InstantiationAwareBeanPostProcessorAdapter的描述,有需要的可以查看我之前的文章,其中比較詳細描述到Spring中的InstantiationAwareBeanPostProcessor類的作用。
步驟四:
對應的業(yè)務類的實現(xiàn)如下:
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc
*/
@Slf4j
@Service
public class AFuncService implements IAFuncService {
@Override
public void login() {
log.info("[{}]拋出登錄事件 ... ", this.getClass());
EventContainer.submit(LoginEvent.class);
}
}
A業(yè)務類,login會在被調(diào)用的生活拋出LoginEvent事件。
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc
*/
@Service
@Slf4j
public class BFuncService implements IBFuncService {
@ReceiveAnno(clz = LoginEvent.class)
private void doAfterLogin() {
log.info("[{}]監(jiān)聽到登錄事件 ... ", this.getClass());
}
}
/**
* @author xifanxiaxue
* @date 3/31/19
* @desc
*/
@Service
@Slf4j
public class CFuncService implements ICFuncService {
@ReceiveAnno(clz = LoginEvent.class)
private void doAfterLogin() {
log.info("[{}]監(jiān)聽到登錄事件 ... ", this.getClass());
}
}
B和C業(yè)務類的doAfterLogin都分別加了注解 @ReceiveAnno(clz = LoginEvent.class) ,在監(jiān)聽到事件LoginEvent后被觸發(fā)。
為了觸發(fā)方便,我在spring提供的測試類內(nèi)加了實現(xiàn),代碼如下:
@RunWith(SpringRunner.class)
@SpringBootTest
public class EventMechanismApplicationTests {
@Autowired
private AFuncService aFuncService;
@Test
public void contextLoads() {
aFuncService.login();
}
}
可以從中看出啟動該測試類后,會調(diào)用業(yè)務A的login函數(shù),而我們要的效果是B業(yè)務類和C業(yè)務類的doAfterLogin函數(shù)會被自動觸發(fā),那么結果如何呢?
結果打印

我們可以從結果打印中看到,在業(yè)務類A的login函數(shù)觸發(fā)后,業(yè)務類B和業(yè)務類C都監(jiān)聽到了監(jiān)聽到登錄事件,證明該機制正常解決了單對多的行為觸發(fā)問題。
總結
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。
- Springboot基于enable模塊驅動的實現(xiàn)
- 詳解Spring Boot Mysql 版本驅動連接池方案選擇
- Spring Bean管理注解方式代碼實例
- Spring Boot conditional注解用法詳解
- 基于spring@aspect注解的aop實現(xiàn)過程代碼實例
- springboot @WebFilter注解過濾器的實現(xiàn)
- Spring實戰(zhàn)之使用TransactionProxyFactoryBean實現(xiàn)聲明式事務操作示例
- spring如何通過FactoryBean配置Bean
- spring中FactoryBean中的getObject()方法實例解析
- Spring注解驅動擴展原理BeanFactoryPostProcessor
相關文章
詳解Java中的File文件類以及FileDescriptor文件描述類
在Java中File類可以用來新建文件和目錄對象,而FileDescriptor類則被用來表示文件或目錄的可操作性,接下來我們就來詳解Java中的File文件類以及FileDescriptor文件描述類2016-06-06
SpringBoot解決同名類導致的bean名沖突bean name conflicts問題
這篇文章主要介紹了SpringBoot解決同名類導致的bean名沖突bean name conflicts問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-06-06
springSecurity之AuthenticationProvider用法解析
這篇文章主要介紹了springSecurity之AuthenticationProvider用法解析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03
Java使用ThreadLocal實現(xiàn)當前登錄信息的存取功能
ThreadLocal和其他并發(fā)工具一樣,也是用于解決多線程并發(fā)訪問,下這篇文章主要給大家介紹了關于Java使用ThreadLocal實現(xiàn)當前登錄信息的存取功能,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-02-02
request如何獲取body的json數(shù)據(jù)
這篇文章主要介紹了request如何獲取body的json數(shù)據(jù)操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-06-06

