一、需求分析
我们需要进行预约设置,就是设置每一天的体检预约最大数量。客户可以通过微信端在线预约,在线预约时需要选择体检的时间,如果客户选择的时间已经预约满则无法进行预约。
二、 Apache POI简介
Apache POI是用Java编写的免费开源的跨平台的Java API,Apache POI提供API给Java程序对Microsoft Office格式档案读和写的功能,其中使用最多的就是使用POI操作Excel文件。
POI结构
HSSF - 提供读写Microsoft Excel XLS格式档案的功能
XSSF - 提供读写Microsoft Excel OOXML XLSX格式档案的功能
HWPF - 提供读写Microsoft Word DOC格式档案的功能
HSLF - 提供读写Microsoft PowerPoint格式档案的功能
HDGF - 提供读Microsoft Visio格式档案的功能
HPBF - 提供读Microsoft Publisher格式档案的功能
HSMF - 提供读Microsoft Outlook格式档案的功能
2.1从Excel文件读取数据
使用POI可以从一个已经存在的Excel文件中读取数据,通过遍历工作表获得行,遍历行获得单元格,最终获取单元格中的值。
//创建工作簿 XSSFWorkbook workbook = new XSSFWorkbook("D:\\test.xlsx"); //获取工作表,既可以根据工作表的顺序获取,也可以根据工作表的名称获取 XSSFSheet sheet = workbook.getSheetAt(0); //遍历工作表获得行对象 for (Row row : sheet) { //遍历行对象获取单元格对象 for (Cell cell : row) { //获得单元格中的值 String value = cell.getStringCellValue(); System.out.println(value); } } workbook.close();
POI操作Excel表格封装了几个核心对象:
XSSFWorkbook:工作簿
XSSFSheet:工作表
Row:行
Cell:单元格
还有一种读取方式就是获取工作表最后一个行号,从而根据行号获得行对象,通过行获取最后一个单元格索引,从而根据单元格索引获取每行的一个单元格对象,代码如下:
//创建工作簿 XSSFWorkbook workbook = new XSSFWorkbook("D:\\test.xlsx"); //获取工作表,既可以根据工作表的顺序获取,也可以根据工作表的名称获取 XSSFSheet sheet = workbook.getSheetAt(0); //获取当前工作表最后一行的行号,行号从0开始 int lastRowNum = sheet.getLastRowNum(); for(int i=0;i<=lastRowNum;i++){ //根据行号获取行对象 XSSFRow row = sheet.getRow(i); short lastCellNum = row.getLastCellNum(); for(short j=0;j<lastCellNum;j++){ String value = row.getCell(j).getStringCellValue(); System.out.println(value); } } workbook.close();
2.2 向Excel文件写入数据
使用POI可以在内存中创建一个Excel文件并将数据写入到这个文件,最后通过输出流将内存中的Excel文件下载到磁盘
//在内存中创建一个Excel文件 XSSFWorkbook workbook = new XSSFWorkbook(); //创建工作表,指定工作表名称 XSSFSheet sheet = workbook.createSheet("测试"); //创建行,0表示第一行 XSSFRow row = sheet.createRow(0); //创建单元格,0表示第一个单元格 row.createCell(0).setCellValue("编号"); row.createCell(1).setCellValue("名称"); row.createCell(2).setCellValue("年龄"); XSSFRow row1 = sheet.createRow(1); row1.createCell(0).setCellValue("1"); row1.createCell(1).setCellValue("小明"); row1.createCell(2).setCellValue("10"); XSSFRow row2 = sheet.createRow(2); row2.createCell(0).setCellValue("2"); row2.createCell(1).setCellValue("小王"); row2.createCell(2).setCellValue("20"); //通过输出流将workbook对象下载到磁盘 FileOutputStream out = new FileOutputStream("D:\\test.xlsx"); workbook.write(out); out.flush(); out.close(); workbook.close();
三、代码实现
批量导入预约设置信息操作过程:
1、点击模板下载按钮下载Excel模板文件
2、将预约设置信息录入到模板文件中
3、点击上传文件按钮将录入完信息的模板文件上传到服务器
4、通过POI读取上传文件的数据并保存到数据库
预约设置信息对应的数据表为t_ordersetting,t_ordersetting表结构:
orderDate:预约日期
number:可预约人数
reservations:已预约人数
步骤一: 提供模板文件
将Excel模板文件放在backend工程的template目录下
步骤二: 实现模板文件下载
为模板下载按钮绑定事件实现模板文件下载
html代码:
<el-card class="box-card"> <div class="boxMain"> <el-button style="margin-bottom: 20px;margin-right: 20px" type="primary" @click="downloadTemplate()">模板下载</el-button> <el-upload action="/ordersetting/upload.do" name="excelFile" :show-file-list="false" :on-success="handleSuccess" :before-upload="beforeUpload"> <el-button type="primary">上传文件</el-button> </el-upload> </div> <div> 操作说明:请点击"模板下载"按钮获取模板文件,在模板文件中录入预约设置数据后点击"上传文件"按钮上传模板文件。 </div> </el-card>
javascript代码:
//上传之前进行文件格式校验 beforeUpload(file){ const isXLS = file.type === 'application/vnd.ms-excel'; if(isXLS){ return true; } const isXLSX = file.type === 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'; if (isXLSX) { return true; } this.$message.error('上传文件只能是xls或者xlsx格式!'); return false; }, //下载模板文件 downloadTemplate(){ window.location.href="../../template/ordersetting_template.xlsx"; }, //上传成功提示 handleSuccess(response, file) { if(response.flag){ this.$message({ message: response.message, type: 'success' }); }else{ this.$message.error(response.message); } console.log(response, file, fileList); },
步骤三: 文件上传
使用ElementUI的上传组件实现文件上传并绑定相关事件
html代码:
<el-card class="box-card"> <div class="boxMain"> <el-button style="margin-bottom: 20px;margin-right: 20px" type="primary" @click="downloadTemplate()">模板下载</el-button> <el-upload action="/ordersetting/upload.do" name="excelFile" :show-file-list="false" :on-success="handleSuccess" :before-upload="beforeUpload"> <el-button type="primary">上传文件</el-button> </el-upload> </div> <div> 操作说明:请点击"模板下载"按钮获取模板文件,在模板文件中录入预约设置数据后点击"上传文件"按钮上传模板文件。 </div> </el-card>
javascript代码:
//上传之前进行文件格式校验 beforeUpload(file){ const isXLS = file.type === 'application/vnd.ms-excel'; if(isXLS){ return true; } const isXLSX = file.type === 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'; if (isXLSX) { return true; } this.$message.error('上传文件只能是xls或者xlsx格式!'); return false; }, //下载模板文件 downloadTemplate(){ window.location.href="../../template/ordersetting_template.xlsx"; }, //上传成功提示 handleSuccess(response, file) { if(response.flag){ this.$message({ message: response.message, type: 'success' }); }else{ this.$message.error(response.message); } console.log(response, file, fileList); },
步骤四: Controller
在common工程中,编写POIUtils工具类,在backend工程创建Controller并提供upload方法处理
/** * Excel文件上传,解析文件内容保存到数据库,实现预约设置数据批量导入 * @param excelFile * @return */ @RequestMapping("/upload") public Result upload(@RequestParam("excelFile")MultipartFile excelFile){ try { List<String[]> list = POIUtils.readExcel(excelFile);//使用POI解析表格数据 if(list != null && list.size() > 0){ List<OrderSetting> orderSettingList = new ArrayList<>(); for (String[] strings : list) { OrderSetting orderSetting = new OrderSetting(new Date(strings[0]), Integer.parseInt(strings[1])); orderSettingList.add(orderSetting); } orderSettingService.add(orderSettingList); } } catch (IOException e) { e.printStackTrace(); return new Result(false, MessageConstant.IMPORT_ORDERSETTING_FAIL); } return new Result(true,MessageConstant.IMPORT_ORDERSETTING_SUCCESS); }
步骤五:编写OrderSettingService 服务接口和服务实现类
为了避免重复导入,需要先判断当前日期是否已经进行设置,如果没有进行预约设置直接insert新增设置,如果已经进行预约设置就需要更新设置。
服务接口:
import com.it.pojo.OrderSetting; import java.util.List; import java.util.Map; public interface OrderSettingService { public void add(List<OrderSetting> list); }
服务实现类:
/** * 预约设置服务 */ @Service(interfaceClass = OrderSettingService.class) @Transactional public class OrderSettingServiceImpl implements OrderSettingService { @Autowired private OrderSettingDao orderSettingDao; //批量添加 public void add(List<OrderSetting> list) { if(list != null && list.size() > 0){ for (OrderSetting orderSetting : list) { //检查此数据(日期)是否存在 long count = orderSettingDao.findCountByOrderDate(orderSetting.getOrderDate()); if(count > 0){ //已经存在,执行更新操作 orderSettingDao.editNumberByOrderDate(orderSetting); }else{ //不存在,执行添加操作 orderSettingDao.add(orderSetting); } } } } }
步骤六:编写Dao接口和实现
实现完成,启动项目运行测试批量导入预约设置。