使用Spring Boot AOP处理方法的入参和返回值

 更新时间:2021年08月16日 10:46:45   作者:Code0cean  
这篇文章主要介绍了使用Spring Boot AOP处理方法的入参和返回值,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

前言

IOC和AOP是Spring 中最重要的两个模块。这里练习一下如何使用Spring Boot AOP处理方法的入参和返回值。

Spring AOP的简单介绍:

AOP(Aspect-Oriented Programming)面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP能够将那些与业务⽆关,却为业务模块所共同调⽤的逻辑或责任(例如事务处理、⽇志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于提高系统的可拓展性和可维护性。

Spring AOP就是基于动态代理的,如果要代理的对象,实现了某个接⼝,那么Spring AOP会使⽤JDK代理,去创建代理对象,⽽对于没有实现接⼝的对象,就⽆法使⽤ JDK代理去进⾏代理了,这时候Spring AOP会使⽤Cglib ,这时候Spring AOP会使⽤ Cglib代理 ⽣成⼀个被代理对象的⼦类来作为代理,如下图所示:

一篇详细介绍AOP的文章:细说Spring——AOP详解(AOP概览)

1. 需求场景

前段时间实习,遇到了一个需求是这样的:项目上线前,项目经理要求有一个用户私密信息的字段需要在数据库中加密存储,从数据库读取出来后需要解密,正常显示到用户界面中。

下面的DEMO中,模拟场景项目经理突然觉得这个用户的身份证号是用户隐私需要进行加密保存,保护用户的隐私,

User类定义如下:

public class User {
    private Integer id;
    private String username;
    private String password;
    private String identityNum;
    //省略getter、setter、toString方法
}

2. 解决方案

因为是临时加的需求,考虑到多个实体类中都会有identityNum属性,为了不侵入原本的业务代码和数据处理代码和业务代码的解耦,一个比较好的方案是使用Spring AOP处理,以DAO层方法做切点,处理字段的加密解密。

3. 代码实现

下面使用Spring Boot+MyBatis实现DEMO,模拟上述场景和解决方案实现。

Controller层UserController类的代码:

@RestController
@RequestMapping("/users")
public class UserController {
    @Autowired
    UserService userService;
    @GetMapping
    public List<User> getAllUsers(){
        return userService.getAllUsers();
    }
    @PostMapping
    public void save(@RequestBody User user){
        userService.save(user);
    }
}

Service层UserService类代码:

@Service
public class UserService {
    @Autowired
    UserDao userDao;
    public List<User> getAllUsers() {
        return userDao.getAllUsers();
    }
    public void save(User user) {
        userDao.save(user);
    }
}

Dao层UserDao接口实现:

@Mapper
public interface UserDao {
    List<User> getAllUsers();
    void save(@Param("user") User user);
}

UserMapper.xml文件实现:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="top.javahai.springbootdemo.dao.UserDao">
    <insert id="save">
        insert into user values (#{user.id},#{user.username},#{user.password},#{user.identityNum})
    </insert>
    <select id="getAllUsers" resultType="top.javahai.springbootdemo.entity.User">
        select id,username,password,identity_num as identityNum  from user
    </select>
</mapper>

切面类UserInfoHandler实现如下,这里只是使用字符串截取的方法模拟加密代码

使用环绕通知@Around注解实现

@Aspect
@Component
public class UserInfoHandler {
    @Pointcut("execution(* top.javahai.springbootdemo.dao.UserDao.*(..))")
    public void pointcut(){
    }
    @Around("pointcut()")
    public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable {
        //处理方法参数,如果是User就进行加密处理
        Object[] args = joinPoint.getArgs();
        for (Object arg : args) {
            if (arg instanceof List){
                if (((List) arg).get(0) instanceof User){
                    ((List<User>) arg).forEach(user->{
                        user.setIdentityNum("encode"+user.getIdentityNum());
                    });
                }
            }
            if (arg instanceof User){
                String identityNum = ((User) arg).getIdentityNum();
                ((User) arg).setIdentityNum("encode"+identityNum);
            }
        }
        //执行方法,获取返回值
        Object obj = joinPoint.proceed();
        //处理方法返回值
        if (obj instanceof List){
            if (!((List) obj).isEmpty()){
                if (((List) obj).get(0) instanceof User){
                    ((List<User>) obj).forEach(data->{
                        data.setIdentityNum(data.getIdentityNum().substring(6));
                    });
                }
            }
        }
        return obj;
    }
}

如果是在其他实体类中也存在identityNum身份证字段,则需要在@PointCut中定义多个切点,另外处理的地方需要添加多个判断。

定义多个切点:

    @Pointcut("execution(* top.javahai.springbootdemo.dao.UserDao.*(..)) ||" +
            "execution(* top.javahai.springbootdemo.dao.ResumeDao.*(..))")
    public void pointcut(){}

4. 测试

通过http://localhost:8080/users接口,将保存一个新的用户数据到数据库中

在这里插入图片描述

查看数据库的存储:

在这里插入图片描述

取出所有的用户数据:

在这里插入图片描述

从测试结果可以看到代码可以正确的处理方法的入参和返回值。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Java正则验证电话,手机,邮箱,日期,金额的方法示例

    Java正则验证电话,手机,邮箱,日期,金额的方法示例

    这篇文章主要介绍了Java正则验证电话,手机,邮箱,日期,金额的方法,结合具体实例形式分析了Java针对电话,手机,邮箱,日期,金额的正则判定操作技巧,需要的朋友可以参考下
    2017-03-03
  • Java和SQL实现取两个字符间的值

    Java和SQL实现取两个字符间的值

    这篇文章主要介绍了Java和SQL实现取两个字符间的值操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-06-06
  • SpringBoot中的跨域详解

    SpringBoot中的跨域详解

    这篇文章主要介绍了SpringBoot中的跨域详解,在浏览器上当前访问的网站,向另一个网站发送请求,用于获取数据的过程就是跨域请求,跨域是浏览器的同源策略决定的,是一个重要的浏览器安全策略,需要的朋友可以参考下
    2023-08-08
  • 一篇文章带你搞定JAVA内存泄漏

    一篇文章带你搞定JAVA内存泄漏

    今天小编就为大家分享一篇关于Java内存泄漏问题处理方法经验总结,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2021-07-07
  • springboot中引入日志文件生成的配置详解

    springboot中引入日志文件生成的配置详解

    本文主要介绍了springboot中引入日志文件生成的配置详解,包括日志级别的设置、日志格式的配置以及日志输出的位置等,从而帮助开发者更好地进行开发与调试
    2023-10-10
  • java实现贪吃蛇游戏代码(附完整源码)

    java实现贪吃蛇游戏代码(附完整源码)

    这篇文章主要介绍了java实现贪吃蛇游戏代码(附完整源码),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-01-01
  • java编译命令基础知识点

    java编译命令基础知识点

    在本篇文章里小编给大家整理的是一篇关于java编译命令基础知识点内容,有兴趣的朋友们可以学习下。
    2021-01-01
  • thymeleaf实现前后端数据交换的示例详解

    thymeleaf实现前后端数据交换的示例详解

    Thymeleaf 是一款用于渲染 XML/XHTML/HTML5 内容的模板引擎,当通过 Web 应用程序访问时,Thymeleaf 会动态地替换掉静态内容,使页面动态显示,这篇文章主要介绍了thymeleaf实现前后端数据交换,需要的朋友可以参考下
    2022-07-07
  • Java使用桥接模式实现开关和电灯照明功能详解

    Java使用桥接模式实现开关和电灯照明功能详解

    这篇文章主要介绍了Java使用桥接模式实现开关和电灯照明功能,较为详细的讲述了桥接模式的概念、原理并结合实例形式分析了Java使用桥接模式实现开关和电灯照明功能相关操作步骤与注意事项,需要的朋友可以参考下
    2018-05-05
  • Java外观模式解读,让你的代码优雅又高效

    Java外观模式解读,让你的代码优雅又高效

    外观模式(Facade Pattern)是一种常用的结构型设计模式,它为复杂的子系统提供一个简单的接口,隐藏复杂的实现细节,本文就来讲讲它是如何简化代码,提高可维护性的
    2023-05-05

最新评论