当前位置:   article > 正文

使用Mybatis的TypeHandler处理数据的加解密_type-handlers-package

type-handlers-package

一、背景

有些项目需要对一些信息入库前进行加密处理,为了数据安全或者隐私合规,但与此同时也使数据处理变得麻烦,不可避免的会带来重复冗长的代码。如果能在持久层处理好数据,避免在业务层处理,就能合理的规避这个问题。

二、方案

使用mybatis框架提供的TypeHandler来实现在持久层处理数据。

TypeHandler简介

Typehandler是mybatis提供的一个接口,通过实现这个接口,可以实现jdbc类型数据和java类型数据的转换,我们常看到的varchar转string、bigint转long等都是mybatis自身实现此接口处理的。

​​​​​​​

 我们可以自己实现一个Typehandler,满足自己的需求。

三、详细实现

1.实现接口,定义自己的Typehandler

一般实现BaseTypeHandler接口即可。笔者的加解密使用了hutool提供的des加密,maven坐标如下:

  1. <dependency>
  2. <groupId>cn.hutool</groupId>
  3. <artifactId>hutool-all</artifactId>
  4. <version>5.8.10</version>
  5. </dependency>

代码如下,可根据业务需求编写方法实现代码:

  1. package com.example.cryptotypehandler.common;
  2. import cn.hutool.crypto.symmetric.SymmetricAlgorithm;
  3. import cn.hutool.crypto.symmetric.SymmetricCrypto;
  4. import lombok.extern.slf4j.Slf4j;
  5. import org.apache.ibatis.type.BaseTypeHandler;
  6. import org.apache.ibatis.type.JdbcType;
  7. import org.apache.ibatis.type.MappedJdbcTypes;
  8. import org.apache.ibatis.type.MappedTypes;
  9. import java.sql.CallableStatement;
  10. import java.sql.PreparedStatement;
  11. import java.sql.ResultSet;
  12. import java.sql.SQLException;
  13. @Slf4j
  14. @MappedJdbcTypes(JdbcType.VARCHAR)
  15. @MappedTypes(String.class)
  16. public class CryptoTypeHandler extends BaseTypeHandler<String> {
  17. private final byte[] key = {-26, -70, -29, -99, 73, -82, 91, -50, 79, -77, 59, 104, 2, -36, 50, -22, -39, -15, -57, -89, 81, -99, 42, -89};
  18. private final SymmetricCrypto des = new SymmetricCrypto(SymmetricAlgorithm.DESede, key);
  19. /*
  20. * 加工入参
  21. */
  22. @Override
  23. public void setNonNullParameter(PreparedStatement ps, int i, String parameter, JdbcType jdbcType) throws SQLException {
  24. if (parameter != null) {
  25. //加密
  26. String encryptHex = des.encryptHex(parameter);
  27. log.info("{} ---加密为---> {}", parameter, encryptHex);
  28. ps.setString(i, encryptHex);
  29. }
  30. }
  31. /*
  32. * 根据列名获取返回结果,可在此方法中加工返回值
  33. */
  34. @Override
  35. public String getNullableResult(ResultSet rs, String columnName) throws SQLException {
  36. String originRes = rs.getString(columnName);
  37. if (originRes != null) {
  38. String res = des.decryptStr(originRes);
  39. log.info("{} ---解密为---> {}", originRes, res);
  40. return res;
  41. }
  42. log.info("结果为空,无需解密");
  43. return null;
  44. }
  45. /*
  46. * 根据列下标获取返回结果,可在此方法中加工返回值
  47. */
  48. @Override
  49. public String getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
  50. String originRes = rs.getString(columnIndex);
  51. if (originRes != null) {
  52. String res = des.decryptStr(originRes);
  53. log.info("{} ---解密为---> {}", originRes, res);
  54. return res;
  55. }
  56. log.info("结果为空,无需解密");
  57. return null;
  58. }
  59. /*
  60. * 根据列下标获取返回结果(存储过程),可在此方法中加工返回值
  61. */
  62. @Override
  63. public String getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
  64. String originRes = cs.getString(columnIndex);
  65. if (originRes != null) {
  66. String res = des.decryptStr(originRes);
  67. log.info("{} ---解密为---> {}", originRes, res);
  68. return res;
  69. }
  70. log.info("结果为空,无需解密");
  71. return null;
  72. }
  73. }

 2.注册自定义的TypeHandler

编写好的TypeHandler需要注册到mybatis中,在application.yml或者application.properties中加入配置:

properties文件:

mybatis.type-handlers-package=com.example.cryptotypehandler.common

yml文件

mybatis:
  type-handlers-package: com.example.cryptotypehandler.common

笔者包结构如下

 3.定义mapper层接口

实体对象类

  1. package com.example.cryptotypehandler.domain;
  2. import lombok.Getter;
  3. import lombok.Setter;
  4. import lombok.ToString;
  5. @Getter
  6. @Setter
  7. @ToString
  8. public class AccountDO {
  9. private Long id;
  10. /**
  11. * 用户名
  12. */
  13. private String userName;
  14. /**
  15. * 密码
  16. */
  17. private String password;
  18. }

和普通的mapper没区别:

  1. package com.example.cryptotypehandler.mapper;
  2. import com.example.cryptotypehandler.domain.AccountDO;
  3. import org.apache.ibatis.annotations.Mapper;
  4. import java.util.List;
  5. /**
  6. * <p>
  7. * Mapper 接口
  8. * </p>
  9. *
  10. * @author shuai.mh
  11. * @since 2022-11-29
  12. */
  13. @Mapper
  14. public interface AccountMapper {
  15. int insertEncrypt(AccountDO accountDO);
  16. List<AccountDO> selectAccount(AccountDO accountDO);
  17. }

4.编写mapper.xml

这边有几个注意点:

  1. 首先看insert语句,我们需要加密的是password字段,因此在password后加上typeHandler=com.example.cryptotypehandler.common.CryptoTypeHandler;此外,#{userName}中,jdbcType=varchar不要填写,因为自定义的typerHandler中加了下面两个注解:@MappedJdbcTypes(JdbcType.VARCHAR)和@MappedTypes(String.class),这两个注解表示JdbcType为varchar是会使用此handler,加了的话,userName也会被加密。
  2. 再看select语句,sql和普通的没有区别,但是resultMap中的password映射加了typeHandler="com.example.cryptotypehandler.common.CryptoTypeHandler",代表此字段在转换成实体对象时会被handler处理,此外其他字段的映射jdbcType保持缺省,如果是varchar也会被处理。
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  3. <mapper namespace="com.example.cryptotypehandler.mapper.AccountMapper">
  4. <!-- 通用查询映射结果 -->
  5. <resultMap id="BaseResultMap" type="com.example.cryptotypehandler.domain.AccountDO">
  6. <id column="id" property="id" />
  7. <result column="user_name" property="userName"/>
  8. <result column="password" property="password" typeHandler="com.example.cryptotypehandler.common.CryptoTypeHandler" />
  9. </resultMap>
  10. <!-- 通用查询结果列 -->
  11. <sql id="Base_Column_List">
  12. id, user_name, password
  13. </sql>
  14. <insert id="insertEncrypt">
  15. insert into account (id, user_name, password)
  16. values (#{id}, #{userName}, #{password, typeHandler=com.example.cryptotypehandler.common.CryptoTypeHandler})
  17. </insert>
  18. <select id="selectAccount" resultMap="BaseResultMap">
  19. select <include refid="Base_Column_List"></include>
  20. from account
  21. where user_name = #{userName}
  22. </select>
  23. </mapper>

5.调用接口,简单测试

①新增用户

 ②查看数据库,密码已被加密

 ③查询用户,查询结果已解密

 感谢阅读,欢迎交流!求关注(〃'▽'〃)

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/菜鸟追梦旅行/article/detail/173082
推荐阅读
相关标签
  

闽ICP备14008679号