`
Everyday都不同
  • 浏览: 723341 次
  • 性别: Icon_minigender_1
  • 来自: 宇宙
社区版块
存档分类
最新评论

Spring MVC环境下用poi技术实现Excel的导入导出

阅读更多

在web项目中,对数据的导入导出是非常实用和常见的,而excel文件则是十分常见的格式。

Excel导入——解析已存在的excel文件,并把里面的数据一一对应,插入到数据库表中,同时在页面上显示出来。(通常数据库里面会有一张字段与该excel表头一一对应的表);

Excel导出——把数据库中的表的数据导出,保存在excel文件中。

 

poi相对于jxl,是更为成熟的excel解析技术。这段时间折腾了几天,终于弄清楚了过程,特此记录。

一些主要的基本概念:poi-3.8-x.jar

HSSFWokbook——对应excel文件的工作薄

HSSFSheet——对应excel文件的表空间

HSSFRow——对应excel文件的表的行

HSSFCell——对应excel文件的表单元格

HSSFCellStyle——对应excel文件的表单元格样式

 

一、Excel导入(需上传已存在的excel文件)

已经存在的excel文件:

 

jsp页面:

<form name="" class="k-form col3" id="M2100F004">
	<label class="k-field-label">文件导入:</label>
        <input class="form-control k-field-file" data-allowblank="false" type="file" id="file" name="excelFile" />
	<button type="button" class="xxx" type="SUBMIT" onclick="importExcel">导入</button>
</form>

 

上传的js:

$.ajaxFileUpload({
	url : "demo/demo-importExcel.json",
	//dataType : 'json',
	secureuri : false,
	fileElementId : 'file',
	success : function(res, status) { //服务器成功响应处理函数
		if (status) {
			//some code
		},
		error : function(res, status, e) {//服务器响应失败处理函数
			alert("导入数据异常:文件导入过程异常。");
		}
	});
}else{
	alert("导入数据异常:系统只支持Excel模板文件导入,请选择正确的模板文件.");
	return;
	}
}

 

对应的后台:

@RequestMapping(value = "/demo/demo-importExcel.json")
	@ResponseBody
	public String importExcel( @RequestParam(value = "excelFile") MultipartFile excelFile,HttpServletRequest request) throws BiffException, IOException, KPromptException{
		if (null == excelFile) {
			result = "模板文件为空,请选择文件";
			return result;
		}
				
//		String path = request.getSession().getServletContext().getRealPath("demo2");
		
		String path = "E:\\demo";
		//容错处理
		File dir = new File(path);
		if(!dir.exists()) {
			dir.mkdirs();
		}
	    String fileName = excelFile.getOriginalFilename();//report.xls
	    String fileName2 = excelFile.getName();//excelFile
	    
	    InputStream fis = excelFile.getInputStream();
	   
	   List<Map<String, Stirng>>  data = ExcelImportUtil..parseExcel(fis);
           //解析到的数据就可以做一些数据库的插入操作了……
	   return "success";
	}

 

重点来了,excel导入我把它封装成了一个工具方法:

public class ExcelImportUtil {
	public static List<Map<String, String>> parseExcel(InputStream fis) {
		List<Map<String, String>> data = new ArrayList<Map<String, String>>();;
		try {
			HSSFWorkbook book = new HSSFWorkbook(fis);
			HSSFSheet sheet = book.getSheetAt(0);
			int firstRow = sheet.getFirstRowNum();
			int lastRow = sheet.getLastRowNum();
			//除去表头和第一行
//			ComnDao dao = SysBeans.getComnDao();
			for(int i = firstRow + 1; i<lastRow+1; i++) {
				Map map = new HashMap();
				
				HSSFRow row = sheet.getRow(i);
				int firstCell = row.getFirstCellNum();
				int lastCell = row.getLastCellNum();
				
				
				for(int j=firstCell; j<lastCell; j++) {
					
					HSSFCell cell2 = sheet.getRow(firstRow + 1).getCell(j);
					String key = cell2.getStringCellValue();
					
					HSSFCell cell = row.getCell(j);
					
					if(cell.getCellType() == HSSFCell.CELL_TYPE_NUMERIC) {
						cell.setCellType(HSSFCell.CELL_TYPE_STRING);
					}
					String val = cell.getStringCellValue();
					
//				System.out.println(val);
					
					if(i == firstRow + 1) {
						break;
					}else{
						map.put(key, val);
						
					}
//				System.out.println(map);
				}
				if(i != firstRow + 1) {
					data.add(map);
					System.out.println(map);
				}
			}
			System.out.println(data);
		} catch (IOException e) {
			e.printStackTrace();
		}
		return data;
	}
}

 

可看到数据库中的结果:



 

注意:

1)poi的api读取excel文件时,最常见的两种cell的数据格式是Numeric和String,所以当是数字格式的时候,需要把它的cell type转成String,否则会出现Cannot get a numeric value from a text cell的错误。对应的,设计相应数据库表最好都是Varchar2格式的。

 

2)需注意需要解析的只是表的数据,所以表的标题应该不在解析范围内;但表头需要解析,他们对应数据库表里的字段;row和cell的位置都是从0开始,

 

二、Excel导出(这里用到了spring的AbstractExcelView)

首先页面:

<a    href="demo/demo-exportExcel.json" data-descript="导出测试">导出</a>

<!-- 或者抽离出一个js-->
var exportExcel = function(){
		var url = "demo/demo-exportExcel.json";
		//window.open(url);
                location.href = url;//具体为啥改用这个,是个坑。。强烈建议这个
            }

 对应的后台:

第一步:excel导出的主要工具类:

public class ExcelExportUtil {
	
	public static HSSFWorkbook generateExcel(List<Map<String, String>> list, String title) {
		HSSFWorkbook book = new HSSFWorkbook();
		try{
			File desFile = new File("d:\\人员表.xls");
			FileOutputStream fos = new FileOutputStream(desFile);
                        HSSFSheet sheet = book.createSheet("Sheet1");
			sheet.autoSizeColumn(1, true);//自适应列宽度
			//样式设置
			HSSFCellStyle style = book.createCellStyle();
			style.setFillForegroundColor(HSSFColor.SKY_BLUE.index);
		      style.setFillPattern(HSSFCellStyle.SOLID_FOREGROUND);
		      style.setBorderBottom(HSSFCellStyle.BORDER_THIN);
		      style.setBorderLeft(HSSFCellStyle.BORDER_THIN);
		      style.setBorderRight(HSSFCellStyle.BORDER_THIN);
		      style.setBorderTop(HSSFCellStyle.BORDER_THIN);
		      style.setAlignment(HSSFCellStyle.ALIGN_CENTER);
		      // 生成一个字体
		      HSSFFont font = book.createFont();
		      font.setColor(HSSFColor.VIOLET.index);
		      font.setFontHeightInPoints((short) 12);
		      font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
		      // 把字体应用到当前的样式
		      style.setFont(font);
			
		      
		      HSSFCellStyle style2 = book.createCellStyle();
			      style2.setFillPattern(HSSFCellStyle.SOLID_FOREGROUND);
			      //设置上下左右边框
			      style2.setBorderBottom(HSSFCellStyle.BORDER_THIN);
			      style2.setBorderLeft(HSSFCellStyle.BORDER_THIN);
			      style2.setBorderRight(HSSFCellStyle.BORDER_THIN);
			      style2.setBorderTop(HSSFCellStyle.BORDER_THIN);
			      style2.setAlignment(HSSFCellStyle.ALIGN_CENTER);
			      
		      //填充表头标题
		      int colSize = list.get(0).entrySet().size();
		      System.out.println("size:" + colSize);
		      //合并单元格供标题使用(表名)
		      sheet.addMergedRegion(new CellRangeAddress(0, 0, 0, colSize-1));
		      HSSFRow firstRow = sheet.createRow(0);//第几行(从0开始)
		      HSSFCell firstCell = firstRow.createCell(0);
		      firstCell.setCellValue(title);
		      firstCell.setCellStyle(style);
		      
		      //填充表头header
		      HSSFRow row = sheet.createRow(1);
		      Set<Entry<String, String>> set = list.get(0).entrySet();
		      List<Entry<String, String>> l = new ArrayList<Map.Entry<String,String>>(set);
		      System.out.println("l:" + l.size());
		      for(int i=0; i< l.size(); i++) {
	    		  String key = l.get(i).getKey();
	    		  System.out.println(key);
	    		  HSSFCell cell = row.createCell(i);
	    		  cell.setCellValue(key);
	    		  cell.setCellStyle(style2);
	    	  }
		      
		      //填充表格内容
		      System.out.println("list:" + list.size());
		      for(int i=0; i<list.size(); i++) {
		    	  HSSFRow row2 = sheet.createRow(i+2);//index:第几行
		    	  Map<String, String> map = list.get(i);
		    	  Set<Entry<String, String>> set2 = map.entrySet();
		    	  List<Entry<String, String>> ll = new ArrayList(set2);
		    	  for(int j=0; j<ll.size(); j++) {
		    		  String val = ll.get(j).getValue();
		    		  HSSFCell cell = row2.createCell(j);//第几列:从0开始
		    		  cell.setCellValue(val);
		    		  cell.setCellStyle(style2);
		    	  }
		      }
		      
//		     book.write(fos); 
//		     fos.close();
		} catch(Exception ex) {
			ex.printStackTrace();
		}
		return book;
	}
	
}

 注意:

1)合并单元格的语法,这里new CellRangeAddress(0, 0, 0, colSize-1)底层为:

 public CellRangeAddress(int firstRow, int lastRow, int firstCol, int lastCol)
  {
    super(firstRow, lastRow, firstCol, lastCol);
  }

 因此表示合并第一行第0-colSize-1列,这常常用于放置表的标题。

2)createRow(int rownum)和createCell(int cellnum)表示创建第几行或第几列,都是从0开始。

 

第二步,在第一步得到HSSFWorkbook的基础上,得到excel文件:涉及的jar包:spring-webmvc-4.x.jar

 

public class ViewExcel extends AbstractExcelView {

	@Override
	protected void buildExcelDocument(Map<String, Object> map,
			HSSFWorkbook book, HttpServletRequest request, HttpServletResponse response)
			throws Exception {
//		  String filename = "人员信息.xls";//设置下载时客户端Excel的名称       
//	        filename = encodeFilename(filename, request);//处理中文文件名    
	        response.setContentType("application/vnd.ms-excel");       
	        response.setHeader("Content-disposition", "attachment;filename=" + filename);       
	        OutputStream ouputStream = response.getOutputStream();       
	        book.write(ouputStream);       
	        ouputStream.flush();       
	        ouputStream.close();   

	}
}

 

第三步,控制层:

 

@RequestMapping(value = "/demo/demo-exportExcel.json")
	@ResponseBody
	public ModelAndView report(ModelMap model, HttpServletRequest request, HttpServletResponse response) {
		 ViewExcel viewExcel = new ViewExcel();      
	      Map obj = null;  
	      System.out.println("response:" + response);
	      //获取数据库表生成的workbook  
	      Map condition = new HashMap();  
              //这里是从数据库里查数据并组装成我们想要的数据结构的过程,略。。
              List<Map<String, String>>  data = dao.xxxx;
	      HSSFWorkbook workbook = ExcelExportUtil.generateExcel(data, "人员信息表");  
	      try {  
	       viewExcel.buildExcelDocument(obj, workbook, request, response);  
	      } catch (Exception e) {  
	    // TODO Auto-generated catch block  
	    	  e.printStackTrace();  
	      }  
	      return new ModelAndView(viewExcel, model);     
	}

 

需注意:下载时是否弹出下载选择框是由浏览器的下载选项设置决定的,并不是代码决定的——特没有节操,害我折腾了N久!!

成果:



 

顺便提一下,做的过程遇到的一些j2se的语法问题:

List<String>转换为String[]的方法:

String[] strs = list.toArray(new String[list.size()])即需指定String[]类型的参数,如果直接这样:String[] strs = (String[]) list.toArray();会出现类型转换的异常:

[Ljava.lang.Object; cannot be cast to [Ljava.lang.String;

 

  • 大小: 6.1 KB
  • 大小: 5.8 KB
4
3
分享到:
评论
6 楼 贝塔ZQ 2017-01-06  
poi实现excel的导入导出就是好复杂啊 还是pageoffice插件比较简单点。导入:document.getElementById("PageOfficeCtrl1").ExcelImportDialog();  导出:document.getElementById("PDFCtrl1").ShowDialog( 3);直接另存到本地就可以。需要的可以到官网上下载示例看看
5 楼 Everyday都不同 2015-08-04  
miles801 写道
好复杂,个人觉得复用性不怎么样...

我的思路:
1. 指定模板(就是设定好了样式的excel文件,利用@title,@name等来指定当前列要显示的是哪个字段)
2. 加载数据(要写入到excel的数据,比如一个list<YourEntity>或者一个map<String,Object>)
3. 利用模板生成一个新的文档

这样模板可以随意指定,在代码中不需要去关心样式问题,这样就只有一个公共的导出类

只需要指定模板+数据就可以任意导出想要的格式/想要的数据....

非常感谢你的建议。。
4 楼 miles801 2015-08-03  
好复杂,个人觉得复用性不怎么样...

我的思路:
1. 指定模板(就是设定好了样式的excel文件,利用@title,@name等来指定当前列要显示的是哪个字段)
2. 加载数据(要写入到excel的数据,比如一个list<YourEntity>或者一个map<String,Object>)
3. 利用模板生成一个新的文档

这样模板可以随意指定,在代码中不需要去关心样式问题,这样就只有一个公共的导出类

只需要指定模板+数据就可以任意导出想要的格式/想要的数据....
3 楼 Everyday都不同 2015-08-03  
lcwen_13 写道
下载貌似重复调用了

额,第一步只是得到HSSFWorkbook,还没有生成excel文件,第二步才是根据HSSFWorkbook得到excel文件的
2 楼 lcwen_13 2015-08-03  
下载貌似重复调用了
1 楼 尚将军 2015-08-02  
写得很好,感谢楼主分享

相关推荐

    Spring3 MVC + POI 实现 Excel与MySQL 的导入导出

    本主题聚焦于如何利用Spring3 MVC框架结合Apache POI库来实现在Java环境中Excel与MySQL数据库之间的数据导入与导出。 Spring3 MVC是Spring框架的一个组件,专门用于构建Web应用程序,它提供了模型-视图-控制器(MVC...

    spring3.2.5 MVC Poi3.9操作excel批量导入

    使用Spring MVC和Apache POI,我们可以构建一个高效、可扩展的系统,实现从Excel文件批量导入数据到数据库,并能将数据库中的数据导出为Excel文件。这样,数据分析师可以轻松地进行数据预处理和分析。 在Maven2的...

    spring mvc easyui-POI导出excel封装源码

    总结来说,"spring mvc easyui-POI导出excel封装源码"项目是将Spring MVC的后端处理能力与EasyUI的前端展示效果以及POI的Excel处理功能相结合,实现了一个功能强大且界面美观的Excel数据导出功能。这个项目对于需要...

    spring3.0 MVC Poi操作excel批量导入数据库和导出数据

    在本主题中,我们将讨论如何利用Spring 3.0 MVC和Apache POI库来处理Excel文件,实现批量导入数据库和导出数据的功能。 Apache POI是一个强大的开源库,专门用于读写Microsoft Office格式的文件,特别是Excel(....

    poi导入导出及spring

    ### POI导入导出及Spring框架综合应用 #### 一、Apache POI简介与核心功能 Apache POI是Apache软件基金会的Jakarta项目中的一个子项目,它为Java程序员提供了一组API,使得他们能够使用Java来操作Microsoft Office...

    SpringMVC POI Excel 生成导出

    "SpringMVC POI Excel 生成导出" SpringMVC 是一个基于 Java 的 Web 框架,POI 是一个 Java 库,用于操作 Microsoft Office 文件格式,Excel 是一个电子表格软件。今天,我们将在 SpringMVC 中使用 POI 生成 Excel ...

    spring MVC 导出excel

    在“spring MVC 导出excel”这个主题中,我们将深入探讨如何利用Spring MVC框架来实现Excel文件的导出功能,这对于数据展示和分析非常有用。 在实际应用中,我们可能需要将数据库中的大量数据导出到Excel文件,以便...

    springmvc+POI的Excel文件导入导出

    总结起来,"SpringMvc+POI的Excel文件导入导出"涉及到的关键技术包括Spring MVC的文件上传处理、Apache POI的Excel读写操作、以及模板填充和导出。通过熟练掌握这些技术,开发者能够高效地处理企业级应用中的Excel...

    SpringMvc 使用poi导入导出Excel

    本篇文章将详细介绍如何在Spring MVC项目中使用Apache POI库来实现Excel的导入和导出。 Apache POI是Apache软件基金会的一个开源项目,专门用于读写Microsoft Office格式的文件,包括Excel。在Java中,POI提供了API...

    spring mvc excel common view

    源码分析可以帮助我们理解如何在Spring MVC环境中集成Excel导出功能,而工具可能指的是Apache POI或其他辅助开发的工具,如IDE插件。 总的来说,Spring MVC Excel common view涉及到的知识点包括: 1. Spring MVC...

    Spring3MVC and POI

    在实际开发中,Spring 3 MVC与Apache POI的结合非常常见,特别是在需要处理大量数据并导出为Excel报表的场景下。例如,你可以创建一个Controller,通过注解路由处理请求,查询数据库获取数据,然后使用POI将数据写入...

    Jxls+Spring MVC实现Excel导出

    "Jxls+Spring MVC实现Excel导出"的主题聚焦于如何利用Jxls库和Spring MVC框架来实现这一功能。Jxls是一个强大的Java库,它扩展了Apache POI,使得在Excel模板上进行编程变得简单,而Spring MVC是Spring框架的一部分...

    Spring框架实现Excel批量导入数据

    本教程将详细讲解如何利用Spring框架实现Excel数据的批量导入,以提高数据录入效率。 首先,我们需要了解Excel文件的结构。Excel是一种常用的电子表格软件,它的数据通常存储在工作簿(Workbook)中,每个工作簿...

    ssh整合poi导入导出Excel

    在本实例中,我们将探讨如何利用SSH框架与Apache POI库协同工作,实现Excel文件的导入和导出功能。Apache POI是一个开源项目,它允许Java应用程序读写Microsoft Office格式的文件,包括Excel。 首先,SSH框架中的...

    基于SSM的POI导入导出Excel实战

    本教程聚焦于如何在SSM框架下使用Apache POI库来实现Excel数据的导入与导出功能,这对于数据分析、报表生成以及大量数据的批量处理具有重要意义。 首先,我们需要理解SSM框架的组成部分: - **Spring**:提供了一...

    ssm+ maven+excel导入导出源码

    在这个“ssm+maven+excel导入导出源码”项目中,我们可以看到开发者提供了实现Excel数据导入导出功能的源代码,这在诸如数据处理、报表生成等场景中非常实用。 1. **Spring框架**:Spring是Java企业级应用的核心...

    spring mvc学习+数据分页+数据导入导出

    POI可以用来读写Excel文件,非常适合在Web应用中实现数据的导入导出功能。例如,你可以创建一个Excel模板,使用POI将数据写入工作表,然后提供给用户下载;同样,用户也可以上传Excel文件,应用通过POI读取数据并将...

    Struts2 poi动态导入导出Excel源码示例

    结合这两个技术,我们可以实现在Web应用中动态导入和导出Excel数据。 首先,让我们深入理解一下如何使用Struts2和POI来实现Excel的导入导出。 1. **Struts2的Action配置**: 在Struts2中,你需要创建一个Action类...

    文件上传下载 excel导入导出 及http 传输

    Spring MVC_Upload这个文件名提示我们,这可能是一个关于如何在Spring MVC框架下实现这些功能的实际代码示例,对于学习和理解这些技术具有很高的参考价值。在实际开发中,理解并掌握这些技术能够提高开发效率,提升...

    Easyui.+.Spring.Mvc导出Excel

    "Easyui Spring Mvc导出Excel"是一个常见的话题,它涉及了三个关键技术:Easyui、Spring MVC以及Excel导出。 Easyui是一个基于jQuery的UI框架,提供了一系列美观且易于使用的组件,如表格、下拉框、对话框等。在Web...

Global site tag (gtag.js) - Google Analytics