SpringAop实操之记录关键业务请求数据
AOP,中文名称,切面。在不影响业务代码情况下,实现想要的功能,是个真炫酷的事。
aop是个好东西,记录日志是必须的。
记录数据也一样的,那么也是可以用aop来实现的,这里借助注解一起解决问题吧。
因为是关键业务,所以意味着不是所有业务,那么自然地就想到了,可以使用过滤的方式,也就是使用注解,如果有注解那么就代表要记录数据,否则不记录。
记录过程需要做什么呢?
首先,因为是记录业务数据,那么我们可以抽象出一个方法出来,也就是 xxx.addRec()。也就是就切面里只需调用记录方法即可。
那么还剩下几个问题,怎样找到这个方法?参数的处理怎么办?需要使用规范限制吗?
找到这个方法,我们可以使用反射调用,当然需要注解进行调用的类和方法名称(如果固定死则无需标注),用反射的好处就是自由,想怎么写就怎么写,要么失败要么成功。但是low!
参数判定,可以根据输入的类,进行if..else.. 判定,然后直接处理相应参数即可。这样可以很自由,想怎么写参数就怎么写参数。但是这样的话,这个切面就和业务代码完全耦合在一起了,还多了n多无谓的判断。很low!
那么问题来了,怎样才不low呢?
几个理论可以借鉴下:
1. 面向接口编程而非面向类编程;
2. 使用枚举值进行注解统一规划;
3. 使用泛型,进行参数上下限处理;
4. 使用模板方法模式封装参数;
5. 使用线程池;
让我们细看下~
面向接口编程,即规定所有业务处理类都实现一个公共的接口,从而使方法不至于凌乱,定义一个清晰的接口,更易于理解含义。各实现类只需关注自己的逻辑即可。
使用枚举值进行注解参数的限定,可以使所有业务操作都在一处进行罗列,且都符合必须的规范。另外,当哪天发现参数无法满足某些需求时,可方便地在该枚举中加入相应参数以完成需求。
使用泛型,将参数限定一定范围内,如要求入库参数必须继承某个基类以实现统一参数处理,也防止了传入任意参数而必须做相应转换的性能消耗。
使用模板方法模式进行参数封装,因入参中都要求继承一个基类,也就是具有共性的参数,所以应具备自动处理公共参数的能力,但是不可越权处理各业务实现的处理,应让实现类有能力自行处理个性化参数(实现类也可以不处理)。交由实现类处理个性化参数时,应使其具有绝对的能力,不应限制其发挥。
使用线程池技术,将额外的工作交给额外的线程,从而使主业务不爱影响。
具体代码撸一遍:
1. 注解开启新篇章,无注解,不工作。
@Component
public class UserTestService { // 添加注解,代表需要进行相应的逻辑处理
@BizRecordTrans(bizType = BizRecordTypeEnum.USER_ADD_FLOW)
public ResponseEntity<Boolean> addUser(UserlDto terminalReq) {
ResponseEntity<Boolean> ret = ResponseBuilder.buildResponse();
// 完成自己的业务...
return ret;
}
}
2. 切面配置,做一个幕后的老兵。
@Component
@Aspect
public class BizRecordAop {
private Logger logger = LoggerFactory.getLogger(this.getClass()); @Around("execution(* com.xxx.dubbo.*.*(..))) and @annotation(com.xxx.spring.annotation.BizRecordTrans)")
public Object deal(ProceedingJoinPoint pjp) throws Throwable {
Object retVal = null;
String methodName = pjp.getSignature().getName();
Object[] args = pjp.getArgs();
Class<?> classTarget = pjp.getTarget().getClass();
Class<?>[] argTypes = ((MethodSignature) pjp.getSignature()).getParameterTypes();
Method objMethod = classTarget.getMethod(methodName, argTypes);
BizRecordTrans bizRecordTrans = objMethod.getAnnotation(BizRecordTrans.class);
BizRecordTypeEnum bizRecordTypeEnum = null;
if (BizRecordTrans != null) {
bizRecordTypeEnum = BizRecordTrans.bizType();
}
try {
retVal = pjp.proceed();
} catch (Exception e) {
// ignore
throw e;
} finally {
if (bizRecordTrans != null) {
Class<? extends BaseEntity> bizRecEntityCls = bizRecordTypeEnum.getBizRecEntityCls();
BaseEntity recEntity = (BaseEntity) bizRecEntityCls.newInstance();
if (args[0] instanceof BaseDto) {
BaseDto baseDto = (BaseDto) args[0];
fillBaseFields(recEntity, baseDto);
}
try {
ThreadPoolTaskExecutor executor = (ThreadPoolTaskExecutor) SpringContextsUtil.getBean("threadPoolTaskExecutor");
executor.execute(() -> {
BizRecBaseService<BaseEntity> recService = (BizRecBaseService<BaseEntity>) SpringContextsUtil
.getBean(bizRecordTypeEnum.getHandlerBeanName());
recService.fitOwnParams(recEntity, args);
recService.addRecord(recEntity);
});
} catch (Exception e) {
logger.error("发生异常", e);
}
}
}
return retVal;
} /**
* 填充公共参数
*/
private void fillBaseFields(BaseEntity baseEntity, BaseDto base) {
baseEntity.setUserId(base.getUserId());
baseEntity.setSeqId(base.getSeqId());
baseEntity.setAddIp(base.getAddIp());
}
}
3. 写一个基础接口,让业务去实现。
public interface BizRecordBaseService<T extends BaseEntity> {
/**
* 执行业务方法
*/
public Integer addRecord(T bizRecord); /**
* 个性化参数填充方法,选择性实现
*/
default void fixOwnParams(T bizRecord, Object[] rawData) {
System.out.println("可以不实现");
}
}
4. 写一个基础类,让所有其他业务实体继承以实现公共参数的封装。
@Data
public class BaseEntity {
/**
* ID
*/
private Long id; /**
* 用户ID
*/
private Long userId; /**
* 请求序列id
*/
private String seqId; /**
* 添加ip
*/
private String addIp; }
5. 打一个组合拳,搞定。
原理浅显,易懂。性能嘛,也ok,无需太担心。
SpringAop实操之记录关键业务请求数据的更多相关文章
- WebAPI获取客户端请求数据
1.什么是WebAPI,详见:http://www.cxyclub.cn/n/25123/2.一般情况下我们不需要去关心客户端的请求数据,WebAPI会通过自己的方式去将客户端请求的数据转换为实体对象 ...
- [Django框架 - 静态文件配置、request对象方法初识、 pycharm链接数据库、ORM实操增删改查、django请求生命周期]
[Django框架 - 静态文件配置.request对象方法初识. pycharm链接数据库.ORM实操增删改查.django请求生命周期] 我们将html文件默认都放在templates文件夹下 将 ...
- springAOP实现操作日志记录,并记录请求参数与编辑前后字段的具体改变
本文为博主原创,未经允许不得转载: 在项目开发已经完成多半的情况下,需要开发进行操作日志功能的开发,由于操作的重要性,需要记录下操作前的参数和请求时的参数, 在网上找了很多,没找到可行的方法.由于操作 ...
- SFUD+FAL+EasyFlash典型场景需求分析,并记一次实操记录
SFUD+FAL+EasyFlash典型场景需求分析:用整个flash存储数据,上千条数据,读取得时候用easyflash很慢,估计要检索整个flash太慢了. 改进方法:分区检索. 1存数据时,根据 ...
- 第7章使用请求测试-测试API . Rspec: everyday-rspec实操。
测试应用与非人类用户的交互,涵盖外部 API 7.1request test vs feature test 对 RSpec 来说,这种专门针 对 API 的测试最好放在 spec/requests ...
- MyBatis实操进阶版(一)
MyBatis实操进阶版(一) 目前而言,持久层框架中,在业务实现灵活性上,无可出MyBatis之右者.具体原因,后续将逐步展开 ResultMap元素的设置 配置文件中,ResultMap元素的作用 ...
- 第十章 Fisco Bcos 权限控制下的数据上链实操演练
一.目的 前面已经完成fisco bcos 相关底层搭建.sdk使用.控制台.webase中间件平台等系列实战开发, 本次进行最后一个部分,体系化管理区块链底层,建立有序的底层控管制度,实现权限化管理 ...
- 干货 | 京东云应用负载均衡(ALB)多功能实操
应用负载均衡(Application Load Balancer,简称ALB)是京东云自主研发的一款七层负载均衡产品,主要面向HTTP和HTTPS流量的WEB应用程序,提供灵活的功能配置.应用负载均衡 ...
- 72 个网络应用安全实操要点,全方位保护 Web 应用的安全
原文地址:Web Application Security Checklist 原文作者:Teo Selenius(已授权) 译者 & 校正:HelloGitHub-小熊熊 & 卤蛋 ...
随机推荐
- dskinlite(uieasy mfc界面库)使用记录4:listbox测试
先看效果图: xml代码: 作者qq:80101277,dskinlite交流qq群:138231653 <window name="listbox1" type=" ...
- python基础之Day17
一.包 1.包 含有--init--.py的文件夹 是模块的一种形式 本质是一个文件夹(用来存放文件,包内所有的文件都是用来被导入的) import 包 包.名字 往包的init里加名字 导 ...
- Linux 远程工具Screen 的应用
挂断原理参考:https://www.ibm.com/developerworks/cn/linux/l-cn-screen/ 要求,python2 常用操作: 创建screen screen -L ...
- Docker 镜像使用
当运行容器是,使用的镜像如果在本地中不存在,docker就会自动从docker镜像仓库中下载,默认是从Docker Hub 公共镜像源下载. 下面我们来学习: 1.管理和使用本地Docker主机镜像 ...
- 当Vue中img的src是动态渲染时不显示问题
最近遇见动态渲染img时,想起了当初刚开始写vue时,曾经遇见的一个小小坑. Vue中:img的src属性是动态渲染时不显示问题1.需求:展示用户头像,数据从后台获取,如果没有拿到则显示默认图片. 如 ...
- Idea如果添加Maven模块
1.要创建一个和heaton-app同级的Maven模块,如果所示 2.点击下一步,添加ArtifactId,其中 groupId : 定义了项目属于哪个组,举个例子,如果你的公司是myc ...
- vue-cli 第一章
一.安装 Node.Python.Git.Ruby 这些都不讲解了 二.安装 Vue-Cli # 最新稳定版本 # 全局安装 npm install --global vue-cli # 创 ...
- canvas简易画板。
在学canvas的时候,想到可以做一个自己用来画画的简易画板,加上canvas的基础都已经学完,便尝试做了一个画板.如图 1.获取标签. var c=document.getElementById(' ...
- win 10 的wordcloud的安装
这两天为了安装wordcloud库可谓是“一把辛酸”,各种出错 jieba什么就不说了,安装和使用都很简单只需要一句代码就可以实现了,而wordcloud在安装之前,本以为也像jieba那样的简单,但 ...
- 绕过D盾的php一句话
d_dun.php <?php $a = '小狗狗你好啊'; $b = '小盾盾你好啊'; foreach ($_REQUEST as $key => $value) { $$key = ...