博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Aspectj 实现Method条件运行
阅读量:5066 次
发布时间:2019-06-12

本文共 2326 字,大约阅读时间需要 7 分钟。

  最近我花了半个小时实现了一个Method的按自定义条件运行的plugin。实现场景是由于我所工作的客户经常会是在同一个代码集上实现多个Brand,所以有些功能只会限制是几个brand调用,而其他的调用则不该调用。还有因为持续交互,我们会不停的release新的功能得到快速的反馈,在这前提下我们会经常遇见在我们刚开发完一个brand的产品代码,就要面临release,所以我们希望其不该对其他的brand产生影响。

  面对这样的需求初级程序员有些人肯定会觉得没什么了不起的啊,不就是几个if/else或者switch/case。我和我的团队对代码有一种天生的洁癖,对于太多复杂的if/elseswitch/case,以及将来会被移除的临时非产品代码分布在各处,有一种抵触心理。

对于有一定工作经验的程序员来说这样的需求肯定也不是什么问题,不就是AOP(面向切面编程),对,这就是我们所期望的解决方案,把处理的逻辑集中,和产品代码的分离。

知道了用AOP,也许你只对了一半,在AOP的世界里,有两种实现方式静态植入和动态代理,最早了AOP实现采用的是静态植入,然后由于IOC之类的框架的兴起,动态代理的实现方式也逐渐兴起,取代了静态植入的方式。但并不是说静态植入的方式就不再有用武之地的,在这里我们所采用的AOP框架Aspectj就是一个静态注入的框架,我们并没有和spring结合,动态代理的方式有个基本的问题就是你不能直接new这个对象这需要交给框架处理,如果是spring框架的话,这要求你必须是一个springbean,以及动态代理会有一些性能的损失。这对于该场景的限制太多,并不是我所期望的。

静态植入的框架在我以前的博客中也提到不少,如果感兴趣请移步 《》。在java世界里Aspectj.net世界里PostSharp(博客中静态植入原理分析篇《》)就是这类框架。

   回到主题,对于 condition-run如何使用请移步

这里简述如何实现:

1 对于AOP第一步是匹配规则,所以定义一个标记:

@Retention(RUNTIME)@Target({METHOD})public @interface ConditionRun {Class
value();}

 

其指向一个实现Runner接口的类型。

2:有了匹配规则,我们就可以找到切入点进行AOP逻辑的处理,

 

@Aspectpublic class ConditionRunAspect {@Around("methodsToBeConditionRun(conditionRun)")public Object profile(ProceedingJoinPoint pjp, ConditionRun conditionRun) throws Throwable {final Result result = isExec(pjp, conditionRun);if (result.isExec()) {return pjp.proceed();}return result.getDefaultValue();}private Result isExec(ProceedingJoinPoint pjp, ConditionRun conditionRun) {try {final Runner runner = conditionRun.value().newInstance();final MethodSignature signature = (MethodSignature) pjp.getSignature();return runner.exec(signature, pjp.getArgs());} catch (Exception e) {throw new RuntimeException("Runner must be empty constructor and make sure the config is ok.", e);}}@Pointcut(value = "@annotation(conditionRun)")public void methodsToBeConditionRun(ConditionRun conditionRun) {}}

 

  这里在切入方法调用之前,new了一个配置的Runner类型(注意必须午餐构造),并调用其exec方法获取是否运行该方法,如果不运行则返回什么默认值。

Runner runner = conditionRun.value().newInstance();final MethodSignature signature = (MethodSignature) pjp.getSignature();return runner.exec(signature, pjp.getArgs());

 

exec的参数签名为方法签名信息和方法运行时参数。

    一切都这么简单,现在你可以任意的框架Runner去做适合你的场景业务了。可以参照项目下得sample实例,该实例展示了一个当出入参数为3的时候执行,部位3则返回默认值1.

   想想还有那些场景,你是否遇见过某类单元测试我只希望运行在某种固定的场景下?假设在开发图形应用的时候,我们希望调用一些不同平台的特定api,虽然我们代码设计封装做得很好,但是我们希望有固定的集成测试去cover这部分逻辑,让我们的测试只运行是固定平台。

转载于:https://www.cnblogs.com/whitewolf/p/3175902.html

你可能感兴趣的文章
多服务器操作利器 - Polysh
查看>>
[LeetCode] Candy
查看>>
Jmeter学习系列----3 配置元件之计数器
查看>>
jQuery 自定义函数
查看>>
jq 杂
查看>>
jquery datagrid 后台获取datatable处理成正确的json字符串
查看>>
作业一
查看>>
AJAX
查看>>
ActiveMQ与spring整合
查看>>
web服务器
查看>>
Git的使用--打tag
查看>>
F# 编程 借助 F# 构建 MVVM 应用程序
查看>>
ACFUN切换代码自用。。。
查看>>
网卡流量检测.py
查看>>
【转】Android的权限permission
查看>>
ajax
查看>>
poj1981 Circle and Points 单位圆覆盖问题
查看>>
POP的Stroke动画
查看>>
线程同步机制初识 【转载】
查看>>
Oracle 游标使用全解
查看>>