文章目录

  • 一、EasyExcel 简介
    • 1、Excel导入导出的应用场景
    • 2、EasyExcel特点
    • 3、案例1:EasyExcel进行Excel写操作
    • 4、案例2:EasyExcel进行Excel读操作
  • 二、课程分类添加功能(后端)
    • 1、引入easyexcel依赖
    • 2、使用代码生成器把课程分类代码生成
    • 3、创建实体类和excel对应关系
    • 4、编写 EduSubjectController 类
    • 5、编写EduSubjectServiceImpl 类
    • 6、创建读取Excel监听器
    • 7、重启oss服务,Swagger中测试文件上传
  • 三、课程分类添加功能(前端)
    • 1、添加课程分类路由
    • 2、创建课程分类页面,修改路由对应的页面路径
    • 3、在添加课程分类页面实现效果
    • 4、启动服务测试
  • 四、课程分类列表(后端)
    • 1、根据返回数据创建对应实体类
    • 2、编写Controller类
    • 3、编写Service类
    • 4、使用swagger进行测试
  • 五、课程分类列表(前端)
    • 1、编写前端接口
    • 2、参考tree模块把前端整合出来
    • 3、前端接口调用
    • 4、启动项目服务测试

一、EasyExcel 简介

1、Excel导入导出的应用场景

  • 数据导入:减轻录入工作量
  • 数据导出:统计信息归档
  • 数据传输:异构系统之间数据传输

2、EasyExcel特点

Java领域解析、生成Excel比较有名的框架有Apache poi、jxl等。但他们都存在一个严重的问题就是非常的耗内存。如果你的系统并发量不大的话可能还行,但是一旦并发上来后一定会OOM或者JVM频繁的full gc。

EasyExcel是阿里巴巴开源的一个excel处理框架,以使用简单、节省内存著称。EasyExcel能大大减少占用内存的主要原因是在解析Excel时没有将文件数据一次性全部加载到内存中,而是从磁盘上一行行读取数据,逐个解析

EasyExcel采用一行一行的解析模式,并将一行的解析结果以观察者的模式通知处理(AnalysisEventListener)。

3、案例1:EasyExcel进行Excel写操作

1、pom中引入xml相关依赖

<dependencies><!-- https://mvnrepository.com/artifact/com.alibaba/easyexcel --><dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>2.1.1</version></dependency><!-- 因为EasyExcel底层是Poi所以需要引入poi的依赖 --><dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId><version>3.17</version></dependency><dependency><groupId>org.apache.poi</groupId><artifactId>poi-ooxml</artifactId><version>3.17</version></dependency>
</dependencies>

2、创建实体类

@Data
public class WriteData {//设置excel表头名称@ExcelProperty("学生编号")private Integer sno;@ExcelProperty("学生姓名")private String sname;
}

3、实现写操作

public static void testWrite(){//实现excel的写操作//1.设置写入文件夹地址和excel名称String filename = "F:\\write.xlsx";//2.调用easyexcel里面的方法实现写操作EasyExcel.write(filename, WriteData.class).sheet("学生列表").doWrite(getData());
}public static List <WriteData> getData(){List <WriteData> list = new ArrayList <>();for(int i = 0;i < 10;i++){String sname = "lucy"+i;list.add(new WriteData(i, sname));}return list;
}

4、案例2:EasyExcel进行Excel读操作

1、创建实体类并标记对应列关系

@Data
public class ReadData {//设置excel表头名称@ExcelProperty(index = 0)private Integer sno;@ExcelProperty(index = 1)private String sname;
}

2、创建监听器进行excel文件读取

public class ExcelListner extends AnalysisEventListener<ReadData> {List<ReadData> list =  new ArrayList<ReadData>();//一行一行的去读取@Overridepublic void invoke(ReadData readData, AnalysisContext analysisContext) {System.out.println("****"+readData);list.add(readData);}//读取Excel表头信息@Overridepublic void invokeHeadMap(Map <Integer, String> headMap, AnalysisContext context) {System.out.println("表头信息:"+headMap);}//读取完成之后执行@Overridepublic void doAfterAllAnalysed(AnalysisContext analysisContext) {}
}

3、进行读操作

public static void testRead(){String filename = "F:\\write.xlsx";EasyExcel.read(filename,ReadData.class,new ExcelListner()).sheet().doRead();
}

二、课程分类添加功能(后端)

核心:EasyExcel读取excel内容实现

1、引入easyexcel依赖

2、使用代码生成器把课程分类代码生成

3、创建实体类和excel对应关系

@Data
public class SubjectData {@ExcelProperty(index = 0)private String oneSujectName;@ExcelProperty(index = 1)private String twoSujectName;
}

4、编写 EduSubjectController 类

@Api(description = "课程分类")
@RestController
@RequestMapping("/eduservice/subject")
@CrossOrigin
public class EduSubjectController {@Autowiredprivate EduSubjectService eduSubjectService;//添加课程分类@ApiOperation(value = "添加课程分类")@PostMapping("addSubjects")public R addSubjects(MultipartFile file){boolean flag = eduSubjectService.saveSubjects(file);if(flag){return R.ok().message("文件导入成功!");}else{return R.error().message("文件导入失败!");}}
}

5、编写EduSubjectServiceImpl 类

@Service
public class EduSubjectServiceImpl extends ServiceImpl<EduSubjectMapper, EduSubject> implements EduSubjectService {@Autowiredprivate EduSubjectService eduSubjectService;//添加课程分类@Overridepublic boolean saveSubjects(MultipartFile file) {try {InputStream inputStream = file.getInputStream();//进行excel文件读取EasyExcel.read(inputStream, SubjectData.class, new SubjectExcelListner(eduSubjectService)).sheet().doRead();return true;}catch (Exception e){e.printStackTrace();return false;}}
}

6、创建读取Excel监听器

public class SubjectExcelListner extends AnalysisEventListener<SubjectData> {//因为SubjectExcelListner在EduSubjectServiceImpl每次会被new的形式使用,所以SubjectExcelListner不能交给//Spring进行管理,也就不能使用@Autowird或@Resource注解注入对象.//但是该类需要调用service中的方法进行数据库操作,该如何使用呢?//可以通过构造方法传递参数的形式,使用service对象.private EduSubjectService eduSubjectService;public SubjectExcelListner(EduSubjectService eduSubjectService) {this.eduSubjectService = eduSubjectService;}public SubjectExcelListner() {}//一行一行的去读取@Overridepublic void invoke(SubjectData subjectData, AnalysisContext analysisContext) {if(subjectData==null){throw new GuLiException(20001,"数据为空!");}//添加一级分类EduSubject oneSubject = this.existOneSubject(subjectData.getOneSubjectName());if(oneSubject==null){//没有相同的一级分类oneSubject = new EduSubject();oneSubject.setTitle(subjectData.getOneSubjectName());oneSubject.setParentId("0");eduSubjectService.save(oneSubject);}//获取一级分类idString pid = oneSubject.getId();//添加二级分类EduSubject twoSubject = this.existTwoSubject(subjectData.getTwoSubjectName(), pid);if(twoSubject==null){twoSubject = new EduSubject();twoSubject.setTitle(subjectData.getTwoSubjectName());twoSubject.setParentId(pid);eduSubjectService.save(twoSubject);}}//读取Excel表头@Overridepublic void invokeHeadMap(Map <Integer, String> headMap, AnalysisContext context) {System.out.println("表头信息:"+headMap);}//读取完成后的操作@Overridepublic void doAfterAllAnalysed(AnalysisContext analysisContext) {}//判断一级分类是否重复private EduSubject existOneSubject(String name){QueryWrapper <EduSubject> wrapper = new QueryWrapper <>();wrapper.eq("title", name).eq("parent_id",0);EduSubject subject = eduSubjectService.getOne(wrapper);return subject;}//判断二级分类是否重复private EduSubject existTwoSubject(String name,String pid){QueryWrapper <EduSubject> wrapper = new QueryWrapper <>();wrapper.eq("title", name).eq("parent_id", pid);EduSubject subject = eduSubjectService.getOne(wrapper);return subject;}
}

7、重启oss服务,Swagger中测试文件上传

http://localhost:8001/swagger-ui.html

三、课程分类添加功能(前端)

1、添加课程分类路由

/src/router/index.js:

{path: '/subject',component: Layout, //布局redirect: '/subject/table',name: 'SubjectSort',meta: { title: '课程分类管理', icon: 'nested' },children: [{path: 'list',name: 'EduSubjectList',component: () =>import ('@/views/edu/subject/list'),meta: { title: '课程分类列表', icon: 'table' }},{path: 'import',name: 'EduSubjectImport',component: () =>import ('@/views/edu/subject/import'),meta: { title: '导入课程分类', icon: 'tree' }}]}

2、创建课程分类页面,修改路由对应的页面路径

3、在添加课程分类页面实现效果

编写页面:

<template><div class="app-container"><el-form label-width="120px"><el-form-item label="信息描述"><el-tag type="info">excel模版说明</el-tag><el-tag><i class="el-icon-download"/><a :href="OSS_PATH + '/excel/template.xlsx'">点击下载模版</a></el-tag></el-form-item><el-form-item label="选择Excel"><el-uploadref="upload":auto-upload="false":on-success="fileUploadSuccess":on-error="fileUploadError":disabled="importBtnDisabled":limit="1":action="BASE_API+'/eduservice/subject/addSubjects'"name="file"accept="application/vnd.ms-excel"><el-button slot="trigger" size="small" type="primary">选取文件</el-button><el-button:loading="false"style="margin-left: 10px;"size="small"type="success"@click="submitUpload">{{ fileUploadBtnText }}</el-button></el-upload></el-form-item></el-form></div>
</template>

编写js方法:

<script>
export default {data() {return {BASE_API: process.env.BASE_API, // 接口API地址OSS_PATH: process.env.OSS_PATH, // 阿里云OSS地址fileUploadBtnText: '上传到服务器', // 按钮文字importBtnDisabled: false, // 按钮是否禁用,loading: false}},created() {},methods: {//上传成功调用的方法fileUploadSuccess(response){if (response.success === true) {this.fileUploadBtnText = '导入成功'this.loading = falsethis.$message({type: 'success',message: response.message})} //跳转到列表页面this.$router.push({path:'/subject/list'})},//上传失败调用的方法fileUploadError(response){if(response.success === false){this.fileUploadBtnText = '导入失败'this.loading = falsethis.$message({type: 'error',message: response.message})}//跳转到列表页面this.$router.push({path:'/subject/list'})},//点击按钮上传文件到接口submitUpload(){this.importBtnDisabled = true //上传按钮禁用this.loading = truethis.$refs.upload.submit()}}
}
</script>

4、启动服务测试

四、课程分类列表(后端)

1、根据返回数据创建对应实体类

//一级分类
@Data
public class OneSubject {private String id;private String title;//一个一级分类里面有多个二级分类private List<TwoSubject> children = new ArrayList<>();
}
//二级分类
@Data
public class TwoSubject {private String id;private String title;
}

返回数据格式为:

2、编写Controller类

@ApiOperation(value = "查询所有课程分类")
@GetMapping("getAllSubject")
public R getAllSubject(){List <OneSubject> oneSubjectList =  eduSubjectService.getAllSubject();return R.ok().data("list",oneSubjectList);
}

3、编写Service类

//查询所有的课程分类==>树形@Overridepublic List <OneSubject> getAllSubject() {//1.查询所有一级分类  parent_id=0QueryWrapper<EduSubject> wrapper = new QueryWrapper<EduSubject>();wrapper.eq("parent_id",0);List <EduSubject> oneSubjectList = this.baseMapper.selectList(wrapper);//2.查询所有二级分类 parent_id!=0wrapper = new QueryWrapper<EduSubject>();wrapper.ne("parent_id",0);List <EduSubject> twoSubjectList = this.baseMapper.selectList(wrapper);//定义最终的返回类型List <OneSubject> finalSubjectList = new ArrayList <OneSubject>();///3 封装一级分类//查询出来所有一级分类list集合集合,得到每一个一级分类对象,回去每一个一级分类对象值,//封装到要求的list集合里面  List<OneSubject> findSubjectList//List<EduSubject> ==> List <OneSubject> finalSubjectListfor (EduSubject subject : oneSubjectList) {//将oneSubjectList封装到finalSubjectListOneSubject oneSubject = new OneSubject();//以下两种方法都可BeanUtils.copyProperties(subject,oneSubject);// oneSubject.setId(subject.getId());// oneSubject.setTitle(subject.getTitle());//在一级分类循环遍历查询所有的二级分类//创建list集合封装每一个一级分类的二级分类List <TwoSubject> children = new ArrayList <>();for (EduSubject eduSubject : twoSubjectList) {TwoSubject twoSubject = new TwoSubject();if (eduSubject.getParentId().equals(subject.getId())){BeanUtils.copyProperties(eduSubject,twoSubject);children.add(twoSubject);}}oneSubject.setChildren(children);finalSubjectList.add(oneSubject);//加入结果集  该语句放在76行之后也可以想想看为啥???}return finalSubjectList;}

4、使用swagger进行测试

五、课程分类列表(前端)

1、编写前端接口

import request from '@/utils/request'export default {//查询所有的课程分类getAllSubject() {return request({url: `/eduservice/subject/getAllSubject`,method: 'get'})}
}

2、参考tree模块把前端整合出来

<template><div class="app-container"><el-input v-model="filterText" placeholder="Filter keyword" style="margin-bottom:30px;" /><el-treeref="tree2":data="data2":props="defaultProps":filter-node-method="filterNode"class="filter-tree"default-expand-all/></div>
</template>

3、前端接口调用

<script>
import subject from '@/api/edu/subject'
export default {data() {return {filterText: '',data2: [],defaultProps: {children: 'children',label: 'title'}}},watch: {filterText(val) {this.$refs.tree2.filter(val)}},created() {//页面加载完成后调用this.getSubjectList()},methods: {getSubjectList(){subject.getAllSubject().then(response=>{this.data2 = response.data.list;})},filterNode(value, data) {if (!value) return truereturn data.title.toLowerCase().indexOf(value) !== -1 //搜索不区分大小写}}
}
</script>

4、启动项目服务测试

谷粒学院(九)EasyExcel | 课程分类模块相关推荐

  1. 谷粒学院——Day07【课程发布-添加课程信息】

    课程分类管理[添加分类前端] 一.配置路由 1. 添加路由 在 src/router/index.js 中添加路由: {// 地址输入path: '/subject',component: Layou ...

  2. 谷粒学院——Day08【课程发布-课程大纲和课程发布】

    富文本编辑器Tinymce 一.Tinymce可视化编辑器 参考 https://panjiachen.gitee.io/vue-element-admin/#/components/tinymce ...

  3. 谷粒学院 —— 10、课程管理:整合阿里云视频点播

    目录 1.创建模块 1.1.引入视频点播所需依赖 1.2.application.properties 1.3.启动类 2.实现视频上传 2.1.后端 2.1.1.创建常量类 2.1.2.上传视频到阿 ...

  4. 2022年最新《谷粒学院开发教程》:4 - 课程管理

    资料 资料地址 后台管理系统目录 前台展示系统目录 1 - 构建工程篇 7 - 渲染前台篇 2 - 前后交互篇 8 - 前台登录篇 3 - 文件上传篇 9 - 前台课程篇 4 - 课程管理篇 10 - ...

  5. 谷粒学院项目笔记6——oss、EasyExcel、课程分类、nginx

    尚硅谷谷粒学院项目第六天内容之对象存储oss服务.使用EasyExcel添加课程分类功能.课程列表分类.ningx的使用 对象存储oss 打开阿里云官网--对象存储--oss管理控制台 创建bucke ...

  6. Day206.课程分类【导入功能】前端实现、 课程分类列表【显示功能】、课程管理【模块需求、添加课程功能】 -谷粒学院

    谷粒学院 [课程分类]-导入前端实现 一.配置路由 1.添加路由 src/router/index.js //课程分类{path: 'subject',component: Layout,redire ...

  7. 2022年最新《谷粒学院开发教程》:9 - 前台课程模块

    资料 资料地址 后台管理系统目录 前台展示系统目录 1 - 构建工程篇 7 - 渲染前台篇 2 - 前后交互篇 8 - 前台登录篇 3 - 文件上传篇 9 - 前台课程篇 4 - 课程管理篇 10 - ...

  8. 谷粒学苑-项目搭建、讲师前后端、课程分类前后端、OSS、EasyExcel

    谷粒学苑 url: jdbc:mysql://localhost:3306/guli?useUnicode=true&useSSL=false&characterEncoding=UT ...

  9. Day215.课程详细页面功能完善、Echarts统计分析模块[生成统计数据+生成图表]前后端整合 -谷粒学院

    谷粒学院 课程详细页面功能完善 一.修改课程详细接口 1.在service_order模块添加接口 用于判断订单中status值是否为1,为1则为已支付 @RestController @CrossO ...

最新文章

  1. Android多进程引发的问题
  2. 《剑指offer》-逐层打印二叉树
  3. conda create出现连接问题_在dockerfile中运行conda更新时出现问题
  4. 作为一名 ABAP 资深顾问,下一步可以选择哪一门 SAP 技术作为主攻方向?
  5. java中怎么判断一段代码时线程安全还是非线程安全_24张图带你彻底理解Java中的21种锁...
  6. 做完四个小项目的收获
  7. python 面试题(3)--- 进制转换
  8. 云上自动化 vs 云上编排
  9. mysql数据库之运行时其他报错
  10. hdu 2545 并查集
  11. web-3g-163(网易)-邮箱和博客-数据架构设计
  12. abaqus 录制结果动画_后处理动画录制
  13. 在VMware ESXi中使用固态硬盘
  14. 基址寻址和变址寻址区别(白话版)
  15. 以自己的电脑作为服务器,搭建网站,外网可访问
  16. MySql 8.0.19安装
  17. 字符编码的故事(ASCII ISO GBK GB2312 UTF-8)
  18. 北斗卫星系统的心脏——中国原子钟发展的历史
  19. Ubuntu自动更换壁纸
  20. react中使用orgchart组织结构插件

热门文章

  1. [PyTorch笔记]数据操作
  2. Android setbackgroundcolor(color.red)color显示找不到符号 符号: 变量 Color
  3. 搜索引擎相关度算法 -BM25 JAVA实现
  4. Notepad++安装NppExec插件运行JAVA和PYTHON程序
  5. 骁龙相机动态设置选项卡
  6. 规范网络游戏 前置审批是关键
  7. 计算机术语中cook,电脑的cookies是什么意思?
  8. 详解c++---红黑二叉树的原理和实现
  9. LeetCode随缘刷题之整数反转
  10. Spreading the Wealth(UVa 11300)