SpringBoot項(xiàng)目導(dǎo)入導(dǎo)出Excel

簡(jiǎn)單呀打、快速的導(dǎo)入導(dǎo)出Excel

一肪虎、安裝依賴

推薦使用最新版本鉴分,可通過文章末尾官方文檔鏈接跳轉(zhuǎn)查看

<dependency>
    <groupId>cn.gjing</groupId>
    <artifactId>tools-excel</artifactId>
    <version>2021.12.3</version>
</dependency>

二哮幢、Excel導(dǎo)出

1、單表頭

定義Excel映射實(shí)體, @Data是lombok的注解

/**
 * @author Gjing
 **/
@Data
@Excel("單級(jí)表頭")
public class SingleHead {
    @ExcelField("姓名")
    private String userName;

    @ExcelField(value = "年齡", format = "0")
    private Integer userAge;

    @ExcelField("性別")
    private Gender gender;

    @ExcelField("愛好")
    private String favorite;
}
/**
 * @author Gjing
 **/
@RestController
public class UserController {

    @GetMapping("/test_export")
    @ApiOperation("導(dǎo)出一級(jí)表頭")
    public void testExport(HttpServletResponse response) {
        //指定映射的實(shí)體為剛剛定義的
        ExcelFactory.createWriter(SingleHead.class, response)
                .write(null)
                .flush();
    }
}
single

2志珍、多級(jí)表頭

數(shù)組中的每個(gè)值代表著一級(jí)表頭

/**
 * @author Gjing
 **/
@Data
@Excel("多級(jí)表頭")
public class MultiHead {
    @ExcelField({"用戶名","用戶名"})
    private String userName;

    @ExcelField({"年齡","年齡"})
    private Integer age;

    @ExcelField({"形態(tài)","身高"})
    private BigDecimal height;

    @ExcelField({"形態(tài)","體重"})
    private BigDecimal weight;
}
/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test_export")
    @ApiOperation("多級(jí)表頭")
    public void testExport(HttpServletResponse response) {
        ExcelFactory.createWriter(MultiHead.class, response)
                //需要在write前激活多級(jí)表頭橙垢,否則不會(huì)自動(dòng)合并
                .multiHead(true)
                .write(null)
                .flush();
    }
}
multi

3、帶大標(biāo)題

大標(biāo)題的起始行是你要插入的sheet中最后一條數(shù)據(jù)的下一行碴裙,如果sheet中沒有數(shù)據(jù)钢悲,就是第一行。你可以配置大標(biāo)題占用的行數(shù)和起始單元格下標(biāo)(默認(rèn)第一個(gè)單元格)和結(jié)束單元格下標(biāo)(默認(rèn)跟隨表頭的數(shù)量)

/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test")
    @ApiOperation("含大標(biāo)題")
    public void testExport(HttpServletResponse response) {
        ExcelFactory.createWriter(SingleHead.class, response)
                //大標(biāo)題占用兩行
                .writeTitle(new BigTitle("我是大標(biāo)題"))
                .write(null)
                .flush();
    }
}
title

4舔株、下拉框

單元格增加下拉框

  • 注解方式
/**
 * @author Gjing
 **/
@Data
@Excel("下拉框?qū)С?)
public class SingleHead {
    @ExcelField("性別")
    @ExcelDropdownBox(combobox = {"男", "女"})
    private Gender gender;

    @ExcelField("愛好")
    private String favorite;
}
/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test_export")
    @ApiOperation("帶下拉框")
    public void testExport(HttpServletResponse response) {
        ExcelFactory.createWriter(SingleHead.class, response)
                //需要在write前激活校驗(yàn)
                .valid(true)
                .write(null)
                .flush();
    }
}
  • 通過方法設(shè)置普通下拉框的選項(xiàng)
/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test_export")
    @ApiOperation("帶下拉框")
    public void testExport(HttpServletResponse response) {
        Map<String, String[]> genderMap = new HashMap<>(8);
        //key為實(shí)體類的字段名莺琳,使用方法進(jìn)行設(shè)置時(shí),實(shí)體字段的@ExcelDropdownBox注解不在需要指定combobox
        //如果指定了也會(huì)去覆蓋注解中的值
        genderMap.put("gender", new String[]{"男", "女"});
        ExcelFactory.createWriter(SingleHead.class, response)
                .valid(true)
                .write(null, genderMap)
                .flush();
    }
}
下拉框

5载慈、時(shí)間校驗(yàn)

導(dǎo)出時(shí)給列表頭下方的單元格增加時(shí)間校驗(yàn)

/**
 * @author Gjing
 **/
@Data
@Excel
public class SingleHead {
    @ExcelField("姓名")
    private String userName;

    @ExcelField(value = "年齡", format = "0")
    private Integer userAge;

    @ExcelField(value = "生日", format = "yyyy-MM-dd")
    @ExcelDateValid(expr1 = "2000-01-01", operatorType = LESS_OR_EQUAL, errorContent = "出生日期不能超過2000年")
    private Date birthday;
}
/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test_export")
    @ApiOperation("帶時(shí)間校驗(yàn)")
    public void testExport(HttpServletResponse response) {
        ExcelFactory.createWriter(SingleHead.class, response)
                //需要在write前激活校驗(yàn)
                .valid(true)
                .write(null)
                .flush();
    }
}
時(shí)間校驗(yàn)

6惭等、數(shù)字、文本校驗(yàn)

導(dǎo)出時(shí)給列表頭下方的單元格增加數(shù)值校驗(yàn)办铡〈亲觯可以對(duì)數(shù)字的大小,文本的長(zhǎng)度進(jìn)行校驗(yàn)

/**
 * @author Gjing
 **/
@Data
@Excel
public class SingleHead {
    @ExcelField("姓名")
    //對(duì)輸入的名稱字?jǐn)?shù)進(jìn)行校驗(yàn)寡具,字?jǐn)?shù)限制小于4
    @ExcelNumericValid(validType = TEXT_LENGTH,operatorType = LESS_THAN,expr1 = "4",errorContent = "姓名字?jǐn)?shù)小于4")
    private String userName;
}
/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test_export")
    @ApiOperation("帶數(shù)值校驗(yàn)")
    public void testExport(HttpServletResponse response) {
        ExcelFactory.createWriter(SingleHead .class, response)
                //需要在write前激活校驗(yàn)
                .valid(true)
                .write(null)
                .flush();
    }
}
數(shù)值校驗(yàn)

7秤茅、重復(fù)校驗(yàn)

導(dǎo)出時(shí)給表頭下方增加數(shù)據(jù)重復(fù)校驗(yàn)

/**
 * @author Gjing
 **/
@Data
@Excel
public class SingleHead {
    @ExcelField("身份證號(hào)")
    @ExcelRepeatValid
    private String idCard;
}
/**
 * @author Gjing
 **/
@RestController
public class TestController {

    @GetMapping("/test_export")
    @ApiOperation("帶數(shù)值校驗(yàn)")
    public void testExport(HttpServletResponse response) {
        ExcelFactory.createWriter(SingleHead .class, response)
                //需要在write前激活校驗(yàn)
                .valid(true)
                .write(null)
                .flush();
    }
}
數(shù)據(jù)重復(fù)

8、數(shù)據(jù)轉(zhuǎn)換

導(dǎo)出時(shí)對(duì)數(shù)據(jù)進(jìn)行加工或者添加默認(rèn)值童叠,支持注解方式和接口方式

  • 注解方式
/**
 * @author Gjing
 **/
@Data
@Excel
public class SingleHead {
    @ExcelField("姓名")
    private String userName;

    @ExcelField(value = "年齡", format = "0")
    //對(duì)每個(gè)人的年齡乘以10
    @ExcelDataConvert(expr1 = "#userAge * 10")
    private Integer userAge;
}
  • 接口方式
/**
 * @author Gjing
 **/
public class MyDataConvert implements DataConvert<SingleHead> {

    @Override
    public Object toEntityAttribute(Object o, Field field) {
        return null;
    }

    @Override
    public Object toExcelAttribute(SingleHead singleHead, Object value, Field field) {
        return (int) value * 10;
    }
}

實(shí)現(xiàn)接口后需要在你需要轉(zhuǎn)換的字段上指定轉(zhuǎn)換器

/**
 * @author Gjing
 **/
@Data
@Excel
public class SingleHead {
    @ExcelField("姓名")
    private String userName;

    @ExcelField(value = "年齡", format = "0", convert = MyDataConvert.class)
    private Integer userAge;
}

導(dǎo)出方法調(diào)用最后一定要使用flush()方法進(jìn)行數(shù)據(jù)刷新到Excel文件中

三框喳、導(dǎo)入

導(dǎo)入的實(shí)體類皆采用導(dǎo)出的實(shí)體類

1、單表頭

/**
 * @author Gjing
 **/
@RestController
public class TestController {
    @Resource
    private UserService userService;
    
    @PostMapping("/user_import")
    @ApiOperation("導(dǎo)入單表頭")
    public void userImport(MultipartFile file) throws IOException {
        ExcelFactory.createReader(file, SingleHead.class)
                //在read()方法前通過訂閱方法增加一個(gè)結(jié)果監(jiān)聽器厦坛,該監(jiān)聽器會(huì)在每一次read()結(jié)束之后觸發(fā)
                .subscribe(e -> this.userService.saveUsers(e))
                .read()
                .finish();
    }
}

2五垮、多級(jí)表頭

前文有提到多級(jí)表頭時(shí),最后一級(jí)為實(shí)際表頭杜秸,所以要在導(dǎo)入時(shí)指定實(shí)際表頭開始下標(biāo)放仗,由于導(dǎo)出的模板映射實(shí)體設(shè)置兩級(jí)表頭,因此這里的實(shí)際表頭為下標(biāo)為1(Excel行和列下標(biāo)都是默認(rèn)0開始的)

/**
 * @author Gjing
 **/
@RestController
public class TestController {
    @Resource
    private UserService userService;
    
    @PostMapping("/user_import")
    @ApiOperation("導(dǎo)入單表頭")
    public void userImport(MultipartFile file) throws IOException {
        ExcelFactory.createReader(file, SingleHead.class)
                //在read()方法前通過訂閱方法增加一個(gè)結(jié)果監(jiān)聽器撬碟,該監(jiān)聽器會(huì)在每一次read()結(jié)束之后觸發(fā)
                //如果Excel中數(shù)據(jù)量太大诞挨,不建議使用結(jié)果監(jiān)聽器莉撇,會(huì)造成生成了過多的映射實(shí)體對(duì)象造成內(nèi)存溢出
                .subscribe(e -> this.userService.saveUsers(e))
                .read(1)
                .finish();
    }
}

在導(dǎo)入調(diào)用結(jié)束后,一定要在最后調(diào)用finish()方法對(duì)流進(jìn)行關(guān)閉


Demo地址:excel-demo
注解參數(shù)說明與更多用法可查看:官方文檔

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末亭姥,一起剝皮案震驚了整個(gè)濱河市稼钩,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌达罗,老刑警劉巖坝撑,帶你破解...
    沈念sama閱讀 221,635評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異粮揉,居然都是意外死亡巡李,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,543評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門扶认,熙熙樓的掌柜王于貴愁眉苦臉地迎上來侨拦,“玉大人,你說我怎么就攤上這事辐宾∮樱” “怎么了?”我有些...
    開封第一講書人閱讀 168,083評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵叠纹,是天一觀的道長(zhǎng)季研。 經(jīng)常有香客問我,道長(zhǎng)誉察,這世上最難降的妖魔是什么与涡? 我笑而不...
    開封第一講書人閱讀 59,640評(píng)論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮持偏,結(jié)果婚禮上驼卖,老公的妹妹穿的比我還像新娘。我一直安慰自己鸿秆,他們只是感情好酌畜,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,640評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著卿叽,像睡著了一般檩奠。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上附帽,一...
    開封第一講書人閱讀 52,262評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音井誉,去河邊找鬼蕉扮。 笑死,一個(gè)胖子當(dāng)著我的面吹牛颗圣,可吹牛的內(nèi)容都是我干的喳钟。 我是一名探鬼主播屁使,決...
    沈念sama閱讀 40,833評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼奔则!你這毒婦竟也來了蛮寂?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,736評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤易茬,失蹤者是張志新(化名)和其女友劉穎酬蹋,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體抽莱,經(jīng)...
    沈念sama閱讀 46,280評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡范抓,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,369評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了食铐。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片匕垫。...
    茶點(diǎn)故事閱讀 40,503評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖虐呻,靈堂內(nèi)的尸體忽然破棺而出象泵,到底是詐尸還是另有隱情,我是刑警寧澤斟叼,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布偶惠,位于F島的核電站,受9級(jí)特大地震影響犁柜,放射性物質(zhì)發(fā)生泄漏洲鸠。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,870評(píng)論 3 333
  • 文/蒙蒙 一馋缅、第九天 我趴在偏房一處隱蔽的房頂上張望扒腕。 院中可真熱鬧,春花似錦萤悴、人聲如沸瘾腰。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,340評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)蹋盆。三九已至,卻和暖如春硝全,著一層夾襖步出監(jiān)牢的瞬間栖雾,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,460評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工伟众, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留析藕,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,909評(píng)論 3 376
  • 正文 我出身青樓凳厢,卻偏偏與公主長(zhǎng)得像账胧,于是被迫代替她去往敵國(guó)和親竞慢。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,512評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容