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

使用Java實現(xiàn)Excel導入并進行數(shù)據(jù)校驗

 更新時間:2025年03月04日 09:59:14   作者:Javaの甘乃迪  
這篇文章主要為大家詳細介紹了Java如何根據(jù)下載的指定數(shù)據(jù)模板寫入數(shù)據(jù)和導入并進行數(shù)據(jù)校驗,文中的示例代碼講解詳細,需要的可以了解下

一、產(chǎn)品需求

1.下載指定的excel數(shù)據(jù)模板

2.excel模板寫入數(shù)據(jù)并導入

3.導入的時候根據(jù)校驗規(guī)則進行篩選,導入成功的返回成功列表,數(shù)據(jù)存在問題的返回失敗列表,失敗列表支持數(shù)據(jù)編輯修正

看到需求的第一眼,可能就覺得第三點有點難度,我們知道,傳統(tǒng)的數(shù)據(jù)校驗可以通過在傳輸對象dto上面加注解實現(xiàn)。

//第一種
public Result test1(@RequestBody @Validated TestDTO dto) {...}
 
//第二種
public Result test2(@RequestBody @Valid TestDTO dto{...}
//第三種
public Result test3(@RequestBody @Validated(value = {SaveGroup.class}) TestDTO dto) {...}
 

TestDTO里面會有一些類似 @NotNull、@NotBlank、@Size等校驗注解,這里就不列了。

然后在全局異常攔截那里進行統(tǒng)一封裝,使其返回的數(shù)據(jù)結構盡量保持統(tǒng)一,所以一般還得有一個RestExceptionHandler類。

@ControllerAdvice
public class RestExceptionHandler {
 
    /**
     * 處理參數(shù)驗證失敗異常
     * @param e
     * @return
     */
    @ExceptionHandler(MethodArgumentNotValidException.class)
    @ResponseBody
    @ResponseStatus(HttpStatus.OK)
    private Response<?> methodArgumentNotValidException(MethodArgumentNotValidException e) {
        log.warn("MethodArgumentNotValidException", e);
        FieldError fieldError = e.getBindingResult().getFieldError();
        return ResponseUtils.create(CommonCodeEnum.VALIDATE_ERROR.getCode(), CommonCodeEnum.VALIDATE_ERROR.getMessage(), fieldError.getDefaultMessage());
    }
}

講到常見的數(shù)據(jù)校驗,那么我們畫風一轉,再回來看需求,可見以上是不滿足需求的,首先,我們的入?yún)⑹且粋€文件流(指定的Excel模板文件),我們得先解析文件再進行數(shù)據(jù)校驗,合法的放一個集合,不合法的放另一個集合;再者,即使入?yún)⑹且粋€數(shù)組,這種校驗一旦不滿足立馬進異常處理了,無法返回給前端正確的數(shù)據(jù)結構,所以今天就分享解決這類需求的解決方案。 

二、解決方法

基礎數(shù)據(jù)

UserExcelVO

import lombok.Data;
 
import java.util.List;
 
/**
 *
 */
@Data
public class UserExcelVO {
 
    /**
     * 成功列表
     */
    private List<UserExcel> success;
 
    /**
     * 失敗列表
     */
    private List<UserExcel> fail;
}

UserExcel

import com.alibaba.excel.annotation.ExcelProperty;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
 
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Pattern;
import javax.validation.constraints.Size;
import java.io.Serializable;
 
/**
 *
 */
@Data
@AllArgsConstructor
@NoArgsConstructor
public class UserExcel implements Serializable {
 
    @NotBlank(message = "手機號不能為空")
    @Size(max = 4)
    @ExcelProperty(value = "用戶名", index = 0)
    private String name;
 
    @ExcelProperty(value = "年齡", index = 1)
    private Integer age;
 
    @Pattern(regexp = "^[1][3,4,5,7,8][0-9]{9}$$", message = "手機號不合法")
    @NotBlank(message = "手機號不能為空")
    @ExcelProperty(value = "手機號", index = 2)
    private String mobile;
 
    @ExcelProperty(value = "性別", index = 3)
    private String sex;
 
}

excel模板數(shù)據(jù)

方案一:大量if-else判斷校驗

import com.alibaba.excel.EasyExcel;
import org.apache.commons.lang3.StringUtils;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;
 
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
 
/**
 *
 */
@RestController
@RequestMapping("/excel")
public class ExcelController {
 
    @PostMapping("/importExcel1")
    public UserExcelVO importExcel(@RequestParam("file") MultipartFile file) {
        List<UserExcel> list;
        List<UserExcel> fail = new ArrayList<>();
        UserExcelVO userExcelVO = new UserExcelVO();
        String mobileReg = "^[1][3,4,5,7,8][0-9]{9}$";
        try {
            list = EasyExcel.read(file.getInputStream(), UserExcel.class, new ModelExcelListener()).sheet().doReadSync();
 
            list.forEach(data -> {
                // 處理姓名的校驗
                if (StringUtils.isEmpty(data.getName()) || data.getName().length() > 4) {
                    fail.add(data);
                    return;
                }
                // 處理手機號的校驗
                if (StringUtils.isEmpty(data.getMobile()) || !data.getMobile().matches(mobileReg)) {
                    fail.add(data);
                    return;
                }
                // 以下根據(jù)字段多少可能有n個if...
 
            });
            userExcelVO.setFail(fail);
            list.removeAll(fail);
            userExcelVO.setSuccess(list);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return userExcelVO;
    }
 
}

方案二:請求體加入注解進行校驗

實際的業(yè)務場景,一個excel里面假如是訂單數(shù)據(jù),最少是幾十個字段起步的,難道要寫幾十個if else嗎?方案一明顯是不合理的,因此使用注解的方式幫我們解決。

ValidationUtils

import javax.validation.Validation;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;
 
/**
 *
 */
public class ValidationUtils {
 
    public static Validator getValidator() {
        return validator;
    }
 
    static Validator validator;
 
    static {
        ValidatorFactory validatorFactory = Validation.buildDefaultValidatorFactory();
        validator = validatorFactory.getValidator();
    }
 
}

ModelExcelListener

import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import lombok.extern.slf4j.Slf4j;
 
import java.util.ArrayList;
import java.util.List;
 
/**
 *
 */
@Slf4j
public class ModelExcelListener extends AnalysisEventListener<UserExcel> {
 
    private List<UserExcel> datas = new ArrayList<>();
 
    /**
     * 通過 AnalysisContext 對象還可以獲取當前 sheet,當前行等數(shù)據(jù)
     */
    @Override
    public void invoke(UserExcel data, AnalysisContext context) {
        //數(shù)據(jù)存儲到list,供批量處理,或后續(xù)自己業(yè)務邏輯處理。
        log.info("讀取到數(shù)據(jù){}",data);
        datas.add(data);
        //根據(jù)業(yè)務自行處理,可以寫入數(shù)據(jù)庫等等
 
    }
 
    //所有的數(shù)據(jù)解析完了調用
    @Override
    public void doAfterAllAnalysed(AnalysisContext context) {
        log.info("所有數(shù)據(jù)解析完成");
    }
 
}

請求

import com.alibaba.excel.EasyExcel;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;
 
import javax.validation.ConstraintViolation;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
 
/**
 *
 */
@RestController
@RequestMapping("/excel")
public class ExcelController {
 
    @PostMapping("/importExcel2")
    public UserExcelVO importExcelV2(@RequestParam("file") MultipartFile file) {
        List<UserExcel> list;
        List<UserExcel> fail = new ArrayList<>();
        UserExcelVO userExcelVO = new UserExcelVO();
        try {
            list = EasyExcel.read(file.getInputStream(), UserExcel.class, new ModelExcelListener()).sheet().doReadSync();
            list.forEach(data -> {
                Set<ConstraintViolation<UserExcel>> violations = ValidationUtils.getValidator().validate(data);
                if (violations.size() > 0) {
                    fail.add(data);
                }
            });
            userExcelVO.setFail(fail);
            list.removeAll(fail);
            userExcelVO.setSuccess(list);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return userExcelVO;
    }
 
}

三、測試結果

方案一的結果:

{
    "success": [
        {
            "name": "張2",
            "age": 19,
            "mobile": "13056781235",
            "sex": "女"
        },
        {
            "name": "張3",
            "age": 20,
            "mobile": "13056781236",
            "sex": "男"
        },
        {
            "name": "張4",
            "age": 21,
            "mobile": "13056781237",
            "sex": "女"
        },
        {
            "name": "張5",
            "age": 22,
            "mobile": "13056781238",
            "sex": "男"
        },
        {
            "name": "張6",
            "age": 23,
            "mobile": "13056781239",
            "sex": "男"
        },
        {
            "name": "張7",
            "age": 24,
            "mobile": "13056781240",
            "sex": "男"
        },
        {
            "name": "張8",
            "age": 25,
            "mobile": "13056781241",
            "sex": "男"
        },
        {
            "name": "張9",
            "age": 26,
            "mobile": "13056781242",
            "sex": "男"
        }
    ],
    "fail": [
        {
            "name": "張1",
            "age": 18,
            "mobile": "3056781234",
            "sex": "男"
        },
        {
            "name": "張10",
            "age": 27,
            "mobile": "130567812436",
            "sex": "男"
        }
    ]
}

方案二的結果:

{
    "success": [
        {
            "name": "張2",
            "age": 19,
            "mobile": "13056781235",
            "sex": "女"
        },
        {
            "name": "張3",
            "age": 20,
            "mobile": "13056781236",
            "sex": "男"
        },
        {
            "name": "張4",
            "age": 21,
            "mobile": "13056781237",
            "sex": "女"
        },
        {
            "name": "張5",
            "age": 22,
            "mobile": "13056781238",
            "sex": "男"
        },
        {
            "name": "張6",
            "age": 23,
            "mobile": "13056781239",
            "sex": "男"
        },
        {
            "name": "張7",
            "age": 24,
            "mobile": "13056781240",
            "sex": "男"
        },
        {
            "name": "張8",
            "age": 25,
            "mobile": "13056781241",
            "sex": "男"
        },
        {
            "name": "張9",
            "age": 26,
            "mobile": "13056781242",
            "sex": "男"
        }
    ],
    "fail": [
        {
            "name": "張1",
            "age": 18,
            "mobile": "3056781234",
            "sex": "男"
        },
        {
            "name": "張10",
            "age": 27,
            "mobile": "130567812436",
            "sex": "男"
        }
    ]
}

發(fā)現(xiàn)兩種方案的測試結果雖然是一樣的,但是很明顯,方案二更優(yōu)秀。我們后續(xù)寫代碼的時候,除了做功能,也要考慮代碼的擴展性,不然產(chǎn)品說加個功能,我們又得吭哧吭哧寫代碼了。

以上就是使用Java實現(xiàn)Excel導入并進行數(shù)據(jù)校驗的詳細內容,更多關于Java Excel導入與數(shù)據(jù)校驗的資料請關注腳本之家其它相關文章!

相關文章

  • mybatis?selectKey賦值未生效的原因分析

    mybatis?selectKey賦值未生效的原因分析

    這篇文章主要介紹了mybatis?selectKey賦值未生效的原因分析,selectKey 會將 SELECT LAST_INSERT_ID()的結果放入到傳入的實體類的主鍵里面,文中通過代碼示例給大家講解非常詳細,需要的朋友可以參考下
    2024-02-02
  • java使用動態(tài)代理來實現(xiàn)AOP(日志記錄)的實例代碼

    java使用動態(tài)代理來實現(xiàn)AOP(日志記錄)的實例代碼

    AOP(面向方面)的思想,就是把項目共同的那部分功能分離開來,比如日志記錄,避免在業(yè)務邏輯里面夾雜著跟業(yè)務邏輯無關的代碼
    2013-09-09
  • Java實現(xiàn)餐廳點餐系統(tǒng)的實例代碼

    Java實現(xiàn)餐廳點餐系統(tǒng)的實例代碼

    這篇文章主要介紹了Java實現(xiàn)餐廳點餐系統(tǒng),本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-06-06
  • Java Spring Security認證與授權及注銷和權限控制篇綜合解析

    Java Spring Security認證與授權及注銷和權限控制篇綜合解析

    Spring Security 是 Spring 家族中的一個安全管理框架,實際上,在 Spring Boot 出現(xiàn)之前,Spring Security 就已經(jīng)發(fā)展了多年了,但是使用的并不多,安全管理這個領域,一直是 Shiro 的天下
    2021-10-10
  • java中實現(xiàn)漢字按照拼音排序(示例代碼)

    java中實現(xiàn)漢字按照拼音排序(示例代碼)

    這篇文章主要是對java中將漢字按照拼音排序的實現(xiàn)代碼進行了詳細的分析介紹。需要的朋友可以過來參考下,希望對大家有所幫助
    2013-12-12
  • druid ParserException類錯誤問題及解決

    druid ParserException類錯誤問題及解決

    這篇文章主要介紹了druid ParserException類錯誤問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • 關于@SpringBootApplication詳解

    關于@SpringBootApplication詳解

    這篇文章主要介紹了關于@SpringBootApplication的使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • @Autowired注解注入的xxxMapper報錯問題及解決

    @Autowired注解注入的xxxMapper報錯問題及解決

    這篇文章主要介紹了@Autowired注解注入的xxxMapper報錯問題及解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • 詳解Spring Security如何在權限中使用通配符

    詳解Spring Security如何在權限中使用通配符

    小伙伴們知道,在Shiro中,默認是支持權限通配符的?,F(xiàn)在給用戶授權的時候,可以一個權限一個權限的配置,也可以直接用通配符。本文將介紹Spring Security如何在權限中使用通配符,需要的可以參考一下
    2022-06-06
  • SpringBoot整合高德地圖天氣查詢的全過程

    SpringBoot整合高德地圖天氣查詢的全過程

    這篇文章主要給大家介紹了關于SpringBoot整合高德地圖天氣查詢的相關資料,文中通過圖文介紹的非常詳細,對大家學習或者使用springboot具有一定的參考學習價值,需要的朋友可以參考下
    2021-12-12

最新評論