1:使用mybatis的BaseTypeHandler类,编写需要加密解密处理的类型

package com.example.poi.typeHandlers;

import cn.hutool.crypto.SecureUtil;
import cn.hutool.crypto.symmetric.AES;
import org.apache.ibatis.type.BaseTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;

import java.nio.charset.StandardCharsets;
import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

/**
 * @Author xu
 * @create 2023/9/3 23
 */
@MappedTypes(EncryptField.class)
@MappedJdbcTypes(JdbcType.VARCHAR)
public class EncryptFieldTypeHandler extends BaseTypeHandler<EncryptField> {
    private static final byte[] KEYS = "balabalbalabala".getBytes(StandardCharsets.UTF_8);

    /**
     * 处理sql执行前相应的参数
     * @param 
     * @param i 参数的索引下标
     * @param parameter
     * @param jdbcType
     * @throws SQLException
     */
    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, EncryptField parameter, JdbcType jdbcType) throws SQLException {
        if (parameter == null || parameter.getValue() == null) {
            ps.setString(i, null);
            return;
        }
        AES aes = SecureUtil.aes(KEYS);
        String encrypt = aes.encryptHex(parameter.getValue());
        ps.setString(i, encrypt);
    }

    /**
     * sql语句执行后执行的操作
     * @param rs
     * @param columnName
     * @return
     * @throws SQLException
     */
    @Override
    public EncryptField getNullableResult(ResultSet rs, String columnName) throws SQLException {
        return decrypt(rs.getString(columnName));
    }

    /**
     * sql语句执行后执行的操作
     * @param rs
     * @param columnIndex
     * @return
     * @throws SQLException
     */
    @Override
    public EncryptField getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        return decrypt(rs.getString(columnIndex));
    }

    /**
     * sql语句执行后执行的操作
     * @param cs
     * @param columnIndex
     * @return
     * @throws SQLException
     */
    @Override
    public EncryptField getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        return decrypt(cs.getString(columnIndex));
    }

    public EncryptField decrypt(String value) {
        if (null == value) {
            return null;
        }
        return new EncryptField(SecureUtil.aes(KEYS).decryptStr(value));
    }
}

只要数据库sql语句有对EncryptField类型的操作,就会触发EncryptFieldTypeHandler类的方法
@MappedTypes:表示该处理器会自动处理的java类型
@MappedJdbcTypes:表示处理器会自动处理的Jdbc类型

2: yml配置需要设置mybatis的type-handlers-package配置

mybatis-plus:
  mapper-locations: classpath*:com/example/poi/mapper/**/xml/*Mapper.xml
  global-config:
    banner: false
    db-config:
      id-type: ASSIGN_ID
      table-underline: true
  type-handlers-package: com.example.poi.typeHandlers

3:传递phone字段是明文,查出结果phone是密文

public class EncryptField{
    private String value;

    public Encrypt() {
    }

    public Encrypt(String value) {
        this.value = value;
    }

    public String getValue() {
        return value;
    }

    public void setValue(String value) {
        this.value = value;
    }
}
    @GetMapping("/getTypeHandler")
    public EntityDemo typeHandler(String phone) {
        EncryptField encryptField = new EncryptField(phone);
        return entityDemoService.getOne(new LambdaQueryWrapper<EntityDemo>().eq(EntityDemo::getPhone, encryptField));
    }

4: 如果要求传递phone字段是明文,查出结果phone是明文

4-1:EntityDemo数据库表对于字段

@Data
@Accessors(chain = true)
@TableName(value = "entity_demo")
public class EntityDemo implements Serializable {
    private static final long serialVersionUID = 1L;

    @TableId
    private Integer id;

    private String age;

    private String name;

    private String phone;
}

4-2:EntityDemoThree查询结果对于字段

@Data
public class EntityDemoThree implements Serializable {
    private static final long serialVersionUID = 1L;

    private Integer id;

    private String age;

    private String name;

    private EncryptField phone;

}
    @GetMapping("/getTypeHandler")
    public EntityDemo typeHandler(String phone) {
        EncryptField encryptField = new EncryptField(phone);
        EntityDemoThree entityDemoThree = entityDemoService.typeHandler3(encryptField);
        EntityDemo entityDemo = new EntityDemo();
        BeanUtils.copyProperties(entityDemoThree, entityDemo);
        entityDemo.setPhone(entityDemoThree.getPhone().getValue());
        return entityDemo;
    }
EntityDemoThree typeHandler(EncryptField phone);
    @Override
    public EntityDemoThree typeHandler(EncryptField phone) {
        return entityDemoMapper.typeHandler3(phone);
    }
    <select id="typeHandler3" resultType="com.example.poi.entity.EntityDemoThree">
        select *
        from entity_demo
        where phone = #{phone}
    </select>

5:对于需要更新的字段最好进行如下条件更新操作

    <update id="refresh">
        UPDATE entity_demo
        <set >
        <if test="age != '' and age != null">
            age = #{age},
        </if>
        <if test="name != '' and name != null">
            name = #{name},
        </if>
        <if test="phone.value != '' and phone.value != null">
            phone = #{phone},
        </if>
        </set>
        WHERE id = #{id}
    </update>

6:如果需要对其他类型进行加密,可以再添加多一个typeHandler类,添加一个List类加密如下:

@MappedJdbcTypes(JdbcType.VARCHAR)
@MappedTypes(List.class)
public class ListTypeHandler extends BaseTypeHandler<List<String>> {
    
    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, List<String> parameter, JdbcType jdbcType) throws SQLException {
        if (null == parameter || parameter.isEmpty()) {
            ps.setString(i, null);
            return;
        }
        ps.setString(i, String.join(",", parameter));
    }
    
    @Override
    public List<String> getNullableResult(ResultSet rs, String columnName) throws SQLException {
        final String value = rs.getString(columnName);
        if (StringUtils.hasText(value)) {
            return Arrays.asList(value.split(","));
        }
        return null;
    }
    
    @Override
    public List<String> getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        final String value = rs.getString(columnIndex);
        if (StringUtils.hasText(value)) {
            return Arrays.asList(value.split(","));
        }
        return null;
    }
    
    @Override
    public List<String> getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        final String value = cs.getString(columnIndex);
        if (StringUtils.hasText(value)) {
            return Arrays.asList(value.split(","));
        }
        return null;
    }
}
Logo

腾讯云面向开发者汇聚海量精品云计算使用和开发经验,营造开放的云计算技术生态圈。

更多推荐