EasyExcel综合课程实战

目录

预热测试

实现写操作,对Excel

对Excel读数据 

项目结合

二级分类

控制层上传Excel文件读取课程

业务层读取Excel

Excel的监听器


场景

当我们导入课程信息,直接导入Excel文件即可录入所有课程信息,它是行模式读取文件信息,节省内存,效率较高;

预热测试


    
    
        com.alibaba
        easyexcel
        2.1.1
    

2.创建实体类

一个是一级

package com.atguigu.demo.excel;

import com.alibaba.excel.annotation.ExcelProperty;
import lombok.Data;

@Data
public class DemoData {
    //设置表头名称
    @ExcelProperty(value = "学生编号",index = 0)
    private Integer sno;

    @ExcelProperty(value = "学生姓名",index = 1)
    private String sname;


}

EasyExcel综合课程实战_第1张图片

实现写操作,对Excel

指定文件名,在文件中write(),并且封装data数据

public static void main(String[] args) throws Exception {
    // 写法1
    String fileName = "F:\\11.xlsx";
    // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
    // 如果这里想使用03 则 传入excelType参数即可
    EasyExcel.write(fileName, DemoData.class).sheet("写入方法一").doWrite(data());
}
//循环设置要添加的数据,最终封装到list集合中
private static List data() {
    List list = new ArrayList();
    for (int i = 0; i < 10; i++) {
        DemoData data = new DemoData();
        data.setSno(i);
        data.setSname("张三"+i);
        list.add(data);
    }
    return list;
}

对Excel读数据 

主要是通过ExcelListener监听器从Excel中获取行数据

package com.atguigu.demo.excel;

import com.alibaba.excel.EasyExcel;

import java.util.ArrayList;
import java.util.List;

public class TestEasyExcel {
    public static void main(String[] args) {
      
        String filename="D:\\write.xlsx";

      
//        EasyExcel.write(filename,DemoData.class).sheet("学生列表").doWrite(getData());

        /**
         * 4.读取excel
         * 读取filename,利用监听器,把每行内容输出
         */
        EasyExcel.read(filename,DemoData.class,new ExcelListener()).sheet().doRead();
    }

    /**
     * 3.返回含有数据的list
     */
    private static List getData(){
        ArrayList list = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
            DemoData data = new DemoData();
            data.setSno(i);
            data.setSname("Lucy"+i);
            list.add(data);
        }
        return list;
    }
}

 读取Excel文件的监听器,监听我们的fileName

package com.atguigu.demo.excel;

import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import org.springframework.stereotype.Component;

import java.util.Map;
@Component
public class ExcelListener extends AnalysisEventListener {

    /**
     * 1.一行一行读取数据
     * @param demoData
     * @param analysisContext
     */
    @Override
    public void invoke(DemoData demoData, AnalysisContext analysisContext) {
        System.out.println("****"+demoData);
    }

    /**
     * 2.去读表头中信息
     * @param
     */
    @Override
    public void invokeHeadMap(Map headMap, AnalysisContext context) {
        System.out.println("表头:"+headMap);
    }

    /**
     * 3.读完之后干的事情
     * @param analysisContext
     */
    @Override
    public void doAfterAllAnalysed(AnalysisContext analysisContext) {

    }
}

项目结合

一级分类

package com.atguigu.eduservice.entity.subject;

import lombok.Data;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;

//一级分类
@Data
public class OneSubject implements Serializable {
    private String id;
    private String title;

    /**
     * 一个一级分类里面有多个二级分类(用集合表示)
     */
    private List children=new ArrayList<>();

}

二级分类

package com.atguigu.eduservice.entity.subject;

import lombok.Data;

import java.io.Serializable;

//二级分类
@Data
public class TwoSubject implements Serializable {
    private String id;
    private String title;

}

控制层上传Excel文件读取课程

@RestController
@RequestMapping("/eduservice/subject")
@CrossOrigin
public class EduSubjectController {

    @Autowired
    private EduSubjectService subjectService;

    /**
     * 1.添加课程分类
     * 获取用户上传过来的文件然后进行读取
     * @Param: MultipartFile
     */
    @PostMapping("addSubject")
    public R addSubject(MultipartFile file){
        //1.1上传过来的excel文件
        subjectService.saveSubject(file,subjectService);

        return R.ok();
    }

}

业务层读取Excel

注意:这里需要注意我们的控制层传入了subjectService,然后业务层中ExcelListener监听器封装了subejectService,需要注意这里出现了相互调用的情况,所以不能都给Spring管理,所以我们没有@AutoWired注入subjectService,而是通过传参的形式

这里再普及一下课程信息展示的流程(树形):

1.首先Wrapper封装父id=0的所有数据,也就是一级分类,然后我们再查询父id!=1的分类也就是二级分类——>2.然后我们对一级分类进行封装,将一级分类转为我们封装的OneSubject——>3.然后我们的封装的一级分类集合遍历,每一个一级分类下有一个二级分类集合属性,将二级分类集合遍历,如果二级id为一级父id,就将当前二级分类加入二级分类集合中——>4.最后一级分类添加这个集合,返回一级分类集合

package com.atguigu.eduservice.service.impl;

import com.alibaba.excel.EasyExcel;
import com.atguigu.eduservice.R;
import com.atguigu.eduservice.entity.EduSubject;
import com.atguigu.eduservice.entity.excel.SubjectData;
import com.atguigu.eduservice.entity.subject.OneSubject;
import com.atguigu.eduservice.entity.subject.TwoSubject;
import com.atguigu.eduservice.listener.SubjectExcelListener;
import com.atguigu.eduservice.mapper.EduSubjectMapper;
import com.atguigu.eduservice.service.EduSubjectService;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Service;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.multipart.MultipartFile;

import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;

/**
 * 

* 课程科目 服务实现类 *

* * @author testjava * @since 2022-07-11 */ @Service public class EduSubjectServiceImpl extends ServiceImpl implements EduSubjectService { /** * 1.添加课程分类,需要读取excel文件 * * @param file 注意这里new的listener,我们传入service目的是自动注入会造成循环依赖,listener调用service,service调用listener,所以不能使用同一个service */ @Override public void saveSubject(MultipartFile file, EduSubjectService subjectService) { try { InputStream in = file.getInputStream(); EasyExcel.read(in, SubjectData.class, new SubjectExcelListener(subjectService)).sheet().doRead(); } catch (Exception e) { e.printStackTrace(); } } /** * 2.获取课程信息 * * @return */ @Override public List getAllOneTwoSubject() { //1.查询所有的一级分类 QueryWrapper wrapper = new QueryWrapper<>(); wrapper.eq("parent_id", "0"); List oneSubjectList = this.list(wrapper); //2.查询所有的二级分类 QueryWrapper wrapperTwo = new QueryWrapper<>(); wrapperTwo.ne("parent_id", "0"); List twoSubjectList = this.list(wrapperTwo); //3.封装所有的一级分类 /** * 查询出来所有的一级分类list集合遍历得到每个一级对象,获取每个一级分类对象值 * 然后封装到List中 */ List finalSubjectList = oneSubjectList.stream().map(eduSubject -> { OneSubject oneSubject = new OneSubject(); BeanUtils.copyProperties(eduSubject, oneSubject); return oneSubject; }).collect(Collectors.toList()); //4.一级分类封装所有的二级分类 for (OneSubject oneSubject : finalSubjectList) { ArrayList twoFinalSubjectList = new ArrayList<>(); for (EduSubject subject : twoSubjectList) { if(subject.getParentId().equals(oneSubject.getId())){ TwoSubject twoSubject = new TwoSubject(); BeanUtils.copyProperties(subject,twoSubject); twoFinalSubjectList.add(twoSubject); } } oneSubject.setChildren(twoFinalSubjectList); } return finalSubjectList; } }

Excel的监听器

 1.首先,是两个分类的判断方法,判断两个分类是否存在,并且返回——>2.如果不为空就返回,如果为空就进行添加——>3.ExcelEasy是一行一行读取,传入SubjectData实体类,里面封装了一级分类和二级分类

package com.atguigu.eduservice.entity.excel;

import com.alibaba.excel.annotation.ExcelProperty;
import lombok.Data;

@Data
public class SubjectData {
   //一级分类和二级分类
    @ExcelProperty(index = 0)
    private String oneSubjectName;

    @ExcelProperty(index = 1)
    private String twoSubjectName;

}
package com.atguigu.eduservice.listener;

import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.atguigu.eduservice.entity.EduSubject;
import com.atguigu.eduservice.entity.excel.SubjectData;
import com.atguigu.eduservice.exceptionHandler.GuliException;
import com.atguigu.eduservice.service.EduSubjectService;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;

/**
 * 通过监听器来进行读和存储
 */
public class SubjectExcelListener extends AnalysisEventListener {

    /**
     * 1.SubjectExcelListener不能交给Spring管理,不然会造成循环依赖
     */
    public EduSubjectService subjectService;

    //手动传subjectService
    public SubjectExcelListener() {}

    public SubjectExcelListener(EduSubjectService subjectService) {
        this.subjectService = subjectService;
    }

    /**
     * 2.监听获取excel中一行行数据
     * @param subjectData
     * @param analysisContext
     */
    @Override
    public void invoke(SubjectData subjectData, AnalysisContext analysisContext) {
        //2.1判断数据是否为空
        if(subjectData==null){
            throw new GuliException(20001,"文件数据为空");
        }

        //2.2一行行读取(一行一列:一级分类),每次读取两个值,一个一级分类,一个二级分类
        EduSubject existOneSubject = this.existOneSubject(subjectService, subjectData.getOneSubjectName());
        if(existOneSubject==null){
            //没有相同的一级分类,我们添加新的一级分类
            existOneSubject = new EduSubject();
            existOneSubject.setParentId("0");
            existOneSubject.setTitle(subjectData.getOneSubjectName());
            subjectService.save(existOneSubject);
        }

        //2.3获取一级分类id
        String pid = existOneSubject.getId();

        //2.4添加二级分类,需要判断二级分类是否重复
        EduSubject existTwoSubject = this.existTwoSubject(subjectService, subjectData.getTwoSubjectName(), pid);
        if(existTwoSubject==null){
            existTwoSubject=new EduSubject();
            existTwoSubject.setParentId(pid);
            existTwoSubject.setTitle(subjectData.getTwoSubjectName());
            subjectService.save(existTwoSubject);
        }
    }

    /**
     * 3.一级title的判断
     * 这里用subjectService目的可能其他地方还需要调用
     * @param
     */
    private EduSubject existOneSubject(EduSubjectService subjectService,String name){
        QueryWrapper wrapper = new QueryWrapper<>();
        wrapper.eq("title",name);
        wrapper.eq("parent_id","0");

        EduSubject oneSubject = subjectService.getOne(wrapper);
        return  oneSubject;
    }

    /**
     * 3.二级分类的判断,得到二级标签
     * @param subjectService
     * @param name:课程名称
     * @param pid:课程号
     */
    private EduSubject existTwoSubject(EduSubjectService subjectService,String name,String pid){
        QueryWrapper wrapper = new QueryWrapper<>();
        wrapper.eq("title",name);
        wrapper.eq("parent_id",pid);

        EduSubject oneSubject = subjectService.getOne(wrapper);
        return oneSubject;
    }




    @Override
    public void doAfterAllAnalysed(AnalysisContext analysisContext) {

    }
}

EasyExcel综合课程实战_第2张图片

 

你可能感兴趣的:(第三方服务,项目,java,数据库,mybatis)