亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

MyBatis利用攔截器實(shí)現(xiàn)數(shù)據(jù)脫敏詳解

 更新時(shí)間:2023年11月01日 15:07:56   作者:小新成長(zhǎng)之路  
現(xiàn)代網(wǎng)絡(luò)環(huán)境中,敏感數(shù)據(jù)的處理是至關(guān)重要的,敏感數(shù)據(jù)包括個(gè)人身份信息、銀行賬號(hào)、手機(jī)號(hào)碼等,所以本文主要為大家詳細(xì)介紹了MyBatis如何利用攔截器實(shí)現(xiàn)數(shù)據(jù)脫敏,希望對(duì)大家有所幫助

背景

現(xiàn)代網(wǎng)絡(luò)環(huán)境中,敏感數(shù)據(jù)的處理是至關(guān)重要的。敏感數(shù)據(jù)包括個(gè)人身份信息、銀行賬號(hào)、手機(jī)號(hào)碼等,泄露這些數(shù)據(jù)可能導(dǎo)致用戶隱私泄露、財(cái)產(chǎn)損失等嚴(yán)重后果。因此,對(duì)敏感數(shù)據(jù)進(jìn)行脫敏處理是一種必要的安全措施。

比如頁(yè)面上常見(jiàn)的敏感數(shù)據(jù)都是加*遮擋處理過(guò)的,如下圖所示。

接下來(lái)本文將以Spring Boot和MyBatis框架實(shí)現(xiàn)返回?cái)?shù)據(jù)的脫敏處理。

脫敏工具

脫敏工具有很多種,本文主要介紹和使用hutool工具包提供的脫敏工具類DesensitizedUtil,它提供了常見(jiàn)的手機(jī)號(hào)、身份證號(hào)、銀行卡、郵箱等脫敏的方法,將敏感數(shù)據(jù)部分加*處理。
使用方法如下:
maven項(xiàng)目需要導(dǎo)入hutool包依賴,坐標(biāo)如下:

<dependency>
    <groupId>cn.hutool</groupId>
    <artifactId>hutool-core</artifactId>
    <version>5.8.4</version>
</dependency>
import cn.hutool.core.util.DesensitizedUtil;
 
public class SensitiveHutoolTest {
    public static void main(String[] args) {
        System.out.println(DesensitizedUtil.mobilePhone("13812345678"));
        System.out.println(DesensitizedUtil.idCardNum("110101200007283706", 3, 4));
        System.out.println(DesensitizedUtil.bankCard("6225809637392380845"));
        System.out.println(DesensitizedUtil.email("zhangsanfeng@test.com"));
    }
}

輸出如下:

138****5678
110***********3706
6225 **** **** *** 0845
z***********@test.com

實(shí)現(xiàn)思路

  • 設(shè)計(jì)一個(gè)常用的脫敏類型枚舉類。
  • 編寫(xiě)脫敏注解,包含脫敏類型,采用注解的方式對(duì)需要脫敏的字段進(jìn)行標(biāo)注。
  • 再編寫(xiě)脫敏注解相應(yīng)的實(shí)現(xiàn)邏輯使用MyBatis的攔截器功能對(duì)查出的結(jié)果集采用反射的處理方式對(duì)結(jié)果進(jìn)行脫敏處理,然后將脫敏處理后的結(jié)果集再返回。

代碼實(shí)現(xiàn)

定義脫敏類型枚舉

利用hutool工具包,對(duì)常見(jiàn)的手機(jī)號(hào)、身份證號(hào)、銀行卡號(hào)、郵箱進(jìn)行脫敏處理。還給了一個(gè)默認(rèn)的DEFAULT枚舉類型按原數(shù)據(jù)返回,同時(shí)自定義了一個(gè)CUSTOM枚舉,可根據(jù)實(shí)現(xiàn)CustomMaskService這個(gè)接口去自定義脫敏邏輯。

注意:這里的CUSTOM只是舉一個(gè)簡(jiǎn)單的例子,實(shí)際情況可能需要根據(jù)實(shí)際情況擴(kuò)展輸入輸出參數(shù)等。

import cn.hutool.core.util.DesensitizedUtil;
import lombok.Getter;
 
public enum SensitiveTypeEnum {
    MOBILE("mobile", "手機(jī)號(hào)") {
        @Override
        public String maskSensitiveData(String data) {
            //手機(jī)號(hào)前3位后4位脫敏,中間部分加*處理,比如:138****5678
            return DesensitizedUtil.mobilePhone(data);
        }
    },
    IDENTIFY("identify", "身份證號(hào)") {
        @Override
        public String maskSensitiveData(String data) {
            //身份證前3位后4位脫敏,中間部分加*處理,比如:110***********3706
            return DesensitizedUtil.idCardNum(data, 3, 4);
        }
    },
    BANKCARD("bankcard", "銀行卡號(hào)") {
        @Override
        public String maskSensitiveData(String data) {
            //銀行卡號(hào)前4位后4位脫敏,中間部分加*處理,比如:6225 **** **** *** 0845
            return DesensitizedUtil.bankCard(data);
        }
    },
 
    EMAIL("email", "郵箱") {
        @Override
        public String maskSensitiveData(String data) {
            //郵箱@符號(hào)后明文顯示,@符號(hào)前的字符串,只顯示第一個(gè)字符,其余加*處理,比如:z***********@test.com
            return DesensitizedUtil.email(data);
        }
    },
    DEFAULT("default", "默認(rèn)") {
        @Override
        public String maskSensitiveData(String data) {
            // 默認(rèn)原值返回,其他這個(gè)也沒(méi)啥意義^_^
            return data;
        }
    },
    CUSTOM("custom", "自定義") {
        @Override
        public String maskSensitiveData(String data, CustomMaskService customMaskService) {
            // 可以自定義處理的service,根據(jù)實(shí)際使用情況可能需要添加參數(shù),調(diào)整一下即可
            return customMaskService.maskData(data);
        }
    };
 
 
    @Getter
    private String type;
 
    @Getter
    private String desc;
 
    SensitiveTypeEnum(String type, String desc) {
        this.type = type;
        this.desc = desc;
    }
 
    /**
     * 遮擋敏感數(shù)據(jù)
     *
     * @param data
     * @return
     */
    public String maskSensitiveData(String data) {
        return data;
    }
 
    public String maskSensitiveData(String data, CustomMaskService customMaskService) {
        return null;
    }
}

定義一個(gè)脫敏注解

import java.lang.annotation.*;
 
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface SensitiveData {
    SensitiveTypeEnum type() default SensitiveTypeEnum.DEFAULT;
}

定義并配置一個(gè)mybatis攔截器。

import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.executor.resultset.ResultSetHandler;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Signature;
import org.springframework.beans.factory.annotation.Autowired;
 
import java.lang.reflect.Field;
import java.sql.Statement;
import java.util.List;
import java.util.Map;
 
@Intercepts({
        @Signature(type = ResultSetHandler.class, method = "handleResultSets", args = {Statement.class})
})
@Slf4j
public class SensitiveDataInterceptor implements Interceptor {
 
    @Autowired
    private CustomMaskService customMaskService;
 
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        Object result = invocation.proceed();
        log.debug("進(jìn)入數(shù)據(jù)脫敏攔截器...");
        if (result instanceof List) {
            List<?> resultList = (List<?>) result;
            for (Object obj : resultList) {
                doSensitiveFields(obj);
            }
        } else if (result instanceof Map) {
            Map<?, ?> resultMap = (Map<?, ?>) result;
            for (Object obj : resultMap.values()) {
                doSensitiveFields(obj);
            }
        } else {
            doSensitiveFields(result);
        }
        return result;
    }
 
    private void doSensitiveFields(Object obj) throws IllegalAccessException {
        Field[] fields = obj.getClass().getDeclaredFields();
        for (Field field : fields) {
            if (field.isAnnotationPresent(SensitiveData.class)) {
                field.setAccessible(true);
                Object value = field.get(obj);
                if (value == null) {
                    return;
                }
                SensitiveData sensitiveData = field.getAnnotation(SensitiveData.class);
                SensitiveTypeEnum type = sensitiveData.type();
                String result;
                if (type == SensitiveTypeEnum.CUSTOM) {
                    result = type.maskSensitiveData(value.toString(), customMaskService);
                } else {
                    result = type.maskSensitiveData(value.toString());
                }
                field.set(obj, result);
            }
        }
    }
}

攔截器注解里寫(xiě)明了要攔截的對(duì)象和方法,類:ResultSetHandler,方法:handleResultSets,它是在sql執(zhí)行完成后拿到結(jié)果集并對(duì)結(jié)果集進(jìn)行處理再返回。

配置mybatis及攔截器

import com.star95.project.study.mybatisplus.interceptor.SensitiveDataInterceptor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
 
@Configuration
@MapperScan("com.star95.project.study.mybatisplus.mapper")
public class MyBatisPlusConfig {
 
    @Bean
    public SensitiveDataInterceptor sensitiveDataInterceptor() {
        return new SensitiveDataInterceptor();
    }
}
 

測(cè)試

import com.star95.project.study.mybatisplus.interceptor.SensitiveData;
import lombok.Data;
 
import static com.star95.project.study.mybatisplus.interceptor.SensitiveTypeEnum.*;
 
@Data
public class UserDto {
    /**
     * id
     */
    private Long id;
    /**
     * 姓名
     */
    @SensitiveData(type = CUSTOM)
    private String name;
    /**
     * 年齡
     */
    private Integer age;
 
    /**
     * 郵箱
     */
    @SensitiveData(type = EMAIL)
    private String email;
 
    /**
     * 手機(jī)號(hào)
     */
    @SensitiveData(type = MOBILE)
    private String mobile;
 
    /**
     * 身份證號(hào)
     */
    @SensitiveData(type = IDENTIFY)
    private String identify;
 
    /**
     * 銀行卡號(hào)
     */
    @SensitiveData(type = BANKCARD)
    private String bankcard;
}
public interface CustomMaskService {
    String maskData(String data);
}
import cn.hutool.core.text.CharSequenceUtil;
import org.springframework.stereotype.Service;
 
@Service
public class CustomMaskServiceImpl implements CustomMaskService {
    @Override
    public String maskData(String data) {
        //第一個(gè)字符明文外,其他都加*處理
        return CharSequenceUtil.hide(data, 1, data.length());
    }
}
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.star95.project.study.mybatisplus.dto.User;
import com.star95.project.study.mybatisplus.dto.UserDto;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;
 
import java.util.List;
 
@Mapper
public interface UserMapper extends BaseMapper<User> {
    @Select({"select * from user where id=#{id}"})
    UserDto getSpecialUser(String id);
 
    @Select({"select * from user"})
    List<UserDto> queryAll();
}
import com.star95.project.study.mybatisplus.dto.UserDto;
import com.star95.project.study.mybatisplus.mapper.UserMapper;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
 
import javax.annotation.Resource;
import java.util.List;
 
@RestController
@RequestMapping("/sensitive")
public class SensitiveMybatisInterceptorTestController {
    @Resource
    private UserMapper userMapper;
 
    @GetMapping("/user/{id}")
    public Result<UserDto> queryUserInfo(@PathVariable String id) {
        return Result.success(userMapper.getSpecialUser(id));
    }
 
    @GetMapping("/userlist")
    public Result<List<UserDto>> queryUserList() {
        return Result.success(userMapper.queryAll());
    }
}

寫(xiě)了兩個(gè)測(cè)試接口,一個(gè)是查詢單個(gè)對(duì)象,一個(gè)是返回list集合列表,訪問(wèn)一下:

可以看到單個(gè)結(jié)果和集合列表都做了脫敏處理,這樣功能就實(shí)現(xiàn)完成了。

其他

使用mybatis攔截器這種方式,是在數(shù)據(jù)持久層的邏輯處理,需要注意的是,查詢結(jié)果返回的dto如果是共享的情況下,可能會(huì)把不需要脫敏的數(shù)據(jù)也給處理了,影響業(yè)務(wù)邏輯,所以使用過(guò)程中要注意區(qū)分。

另外也可在接口返回?cái)?shù)據(jù)時(shí)進(jìn)行脫敏處理,也就是所說(shuō)的序列化方式,比如自定義Jackson、fastjson等的序列化邏輯同樣可以完成數(shù)據(jù)脫敏。

總結(jié)

通過(guò)使用MyBatis攔截器,我們可以實(shí)現(xiàn)對(duì)敏感數(shù)據(jù)的優(yōu)雅脫敏處理,保護(hù)用戶隱私和數(shù)據(jù)安全。這種方式可以靈活應(yīng)用于各種場(chǎng)景,提供了一種簡(jiǎn)單而強(qiáng)大的解決方案。在實(shí)際開(kāi)發(fā)中,我們可以根據(jù)具體需求,定制化開(kāi)發(fā)攔截器的邏輯,以滿足不同的數(shù)據(jù)脫敏需求。

數(shù)據(jù)的隱私和安全是非常重要的,敏感數(shù)據(jù)除存儲(chǔ)方面要加密外,再展示方便也要適當(dāng)?shù)淖雒撁籼幚恚疚闹唤榻B了mybatis攔截器實(shí)現(xiàn)的一種數(shù)據(jù)脫敏方式,還有很多其他技術(shù)可以實(shí)現(xiàn),可以自行搜索,根據(jù)實(shí)際情況選擇合適的解決方案。

以上就是MyBatis利用攔截器實(shí)現(xiàn)數(shù)據(jù)脫敏詳解的詳細(xì)內(nèi)容,更多關(guān)于MyBatis數(shù)據(jù)脫敏的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Mybatis分頁(yè)的實(shí)現(xiàn)及使用注解開(kāi)發(fā)操作

    Mybatis分頁(yè)的實(shí)現(xiàn)及使用注解開(kāi)發(fā)操作

    這篇文章主要介紹了Mybatis分頁(yè)的實(shí)現(xiàn)及使用注解開(kāi)發(fā)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • 如何利用Java使用AOP實(shí)現(xiàn)數(shù)據(jù)字典轉(zhuǎn)換

    如何利用Java使用AOP實(shí)現(xiàn)數(shù)據(jù)字典轉(zhuǎn)換

    這篇文章主要介紹了如何利用Java使用AOP實(shí)現(xiàn)數(shù)據(jù)字典轉(zhuǎn)換,AOP也是我們常說(shuō)的面向切面編程,AOP在我們開(kāi)發(fā)過(guò)程中應(yīng)用也比較多,在這里我們就基于AOP來(lái)實(shí)現(xiàn)一個(gè)數(shù)據(jù)字典轉(zhuǎn)換的案例
    2022-06-06
  • 熟悉maven:使java開(kāi)發(fā)變得更高效

    熟悉maven:使java開(kāi)發(fā)變得更高效

    在日常的開(kāi)發(fā)過(guò)程中,maven是很常見(jiàn)的項(xiàng)目構(gòu)建工具。maven可以極大的提高我們的開(kāi)發(fā)效率,幫助我們簡(jiǎn)化開(kāi)發(fā)過(guò)程中一些解決依賴和項(xiàng)目部署的相關(guān)問(wèn)題,所以學(xué)習(xí)掌握maven的相關(guān)知識(shí)是非常有必要的
    2021-06-06
  • Spring的IOC解決程序耦合的實(shí)現(xiàn)

    Spring的IOC解決程序耦合的實(shí)現(xiàn)

    本文主要介紹了Spring的IOC解決程序耦合的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-07-07
  • Java反射機(jī)制,反射相關(guān)API,反射API使用方式(反射獲取實(shí)體類字段名和注解值)

    Java反射機(jī)制,反射相關(guān)API,反射API使用方式(反射獲取實(shí)體類字段名和注解值)

    這篇文章主要介紹了Java反射機(jī)制,反射相關(guān)API,反射API使用方式(反射獲取實(shí)體類字段名和注解值),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • Java集合教程之Collection實(shí)例詳解

    Java集合教程之Collection實(shí)例詳解

    集合,或者叫容器,是一個(gè)包含多個(gè)元素的對(duì)象,下面這篇文章主要給大家介紹了關(guān)于Java集合教程之Collection的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2018-08-08
  • SpringBoot FreeWorker模板技術(shù)解析

    SpringBoot FreeWorker模板技術(shù)解析

    這篇文章主要介紹了SpringBoot FreeWorker模板技術(shù)解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-11-11
  • IDEA中GitLab的使用詳解

    IDEA中GitLab的使用詳解

    這篇文章主要介紹了IDEA中GitLab的使用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-07-07
  • 使用Spring Boot集成FastDFS的示例代碼

    使用Spring Boot集成FastDFS的示例代碼

    本篇文章主要介紹了使用Spring Boot集成FastDFS的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-02-02
  • 遞歸之斐波那契數(shù)列java的3種方法

    遞歸之斐波那契數(shù)列java的3種方法

    這篇文章主要為大家詳細(xì)介紹了遞歸之斐波那契數(shù)列java的3種方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-02-02

最新評(píng)論