java处理excel的一点实践积累

作者:佟亮  

1   文档说明

   目的:编写本文档主要的目的是与大家分享下如何用java处理excel,文档中的代码一部分来自《企业项目管理系统V2.0》的源码,一部分copy网上的。

     背景:使用Windows操作系统的同事对Excel(电子表格)一定不会陌生,但是要使用Java语言来操纵Excel文件并不是一件容易的事。在Web应用日益盛行的今天,通过Web来操作Excel文件的需求越来越强烈。

2   需求

    本部分与技术无关,没兴趣的同事可以跳过直接看下一章。

   之所以我要把“需求”单列出来讲讲,是因为通过实现这个需要,我学到的不只是技术,更重要的是思想和工作的方式。客户的原始需求是,在web系统实现数据以表格的形式录入,然后对这些数据能够进行公式计算处理(一部分表格里面配置的是公式)。当时接到这个需求的时候,我们二话没说直接用js画表格,js实现公式的计算。结果由于客户不断的按照excel中的功能要求我们实现,我们终于吃不消了,要知道联通的这个需求实际上是excel的全部功能呀。闭门造车的我们殊不知早在几年前就已经有类似的weboffice中间件了。造成我们有这么失误的主要原因是思想的局限和工作方式的死板,所以这是我的一个教训,希望看到文档的同事引以为鉴。之后我们引用了金格公司的weboffice2003这个中间件,其实我们用的只是通过这个中间件将excel在jsp页面打开,然后将excel的内容用他封装的提交方式发送给action,然后我们在action、Bo、dao中处理这个excel,其中如何处理excel就是我接下来要讲的。

3   技术实现

要学习java处理excel,首先大家需要了解jxl这个包,我们引用的所有方法都来自这个包,我的介绍也主要围绕着这个包里的类和方法展开。

3.1   读excel

3.1.1  读取本地excel文件

读取Excel数据表的第一步是创建Workbook(术语:工作薄)。

      InputStream is = new FileInputStream(sourcefile[w1] );

      Workbook wb = Workbook.getWorkbook(is);

创建了Workbook对象,我们就可以通过它来访问Excel Sheet(术语:工作表)。

Sheet rs = rwb.getSheet(0[w2] );

类似这样还可以取到Cell对象,以及cell的相应属性信息。大家可以查看jxl的api文档。 

此外,当你完成对Excel电子表格数据的处理后,一定要使用close()方法来关闭先前创建的对象,以释放读取数据表的过程中所占用的内存空间,在读取大量数据时显得尤为重要。  wb.close();

3.1.2  读取数据库文件

与读取本地excel类似,区别在于输入流的获取。一般excel在数据库以blob类型存储,所以我们在查询的时候要读blob字段,获取输入流。

代码如下:

InputStream instream = vField[w3] .getBinaryStream();

3.2   写excel文件

3.2.1  创建excel文件

3.2.1.1 创建方式

          1、写到本地文件

WritableWorkbook wwb = Workbook.createWorkbook(new File(targetfile[w4] ));

                 2、直接写入到输出流   

ByteArrayOutputStream outs = new ByteArrayOutputStream();
    WritableWorkbook wb = Workbook.createWorkbook(outs);

3.2.1.2 创建步骤

                 1、创建可写的Workbook[w5] (createWorkbook)。

2、创建可写的Sheet[w6] (createSheet)。

3、如果想美化生成的excel可以创建WritableFont,WritableCellFormat来设定颜色,字体等

4、添加Number对象,.添加Boolean对象,添加DateTime对象,添加Label对象及使用mergeCells方法合并单元格。

5、写入Exel工作表 wb.write();

6、关闭Excel工作薄对象wb.close();

详细代码见备注4.1

3.2.2  修改excel文件

3.2.2.1 获取方式

1、本地文件

InputStream is = new FileInputStream(sourcefile[w7] );
ByteArrayOutputStream outs = new ByteArrayOutputStream();

2、数据库对象

InputStream instream = vField[w8] .getBinaryStream();
ByteArrayOutputStream outs = new ByteArrayOutputStream();

 

3.2.2.2 修改步骤

1、创建可写的Workbook[w9] 。

wwbook = Workbook.createWorkbook(outstream, this.rwbook,new WorkbookSettings());

接下来的操作和创建excel一样,返回的都是一个输出流。你可以对这个输出流做你想要的操作。

详细代码见备注4.2

4   备注

4.1   创建excel文件代码举例

//1.添加Label对象
Label labelC = new jxl.write.Label(0, 0, "This is a Label cell");
ws.addCell(labelC);
//添加带有字型Formatting的对象
jxl.write.WritableFont wf = new jxl.write.WritableFont(WritableFont.TIMES, 18, WritableFont.BOLD, true);
jxl.write.WritableCellFormat wcfF = new jxl.write.WritableCellFormat(wf);
jxl.write.Label labelCF = new jxl.write.Label(1, 0, "This is a Label Cell", wcfF);
ws.addCell(labelCF);
//添加带有字体颜色Formatting的对象
jxl.write.WritableFont wfc = new jxl.write.WritableFont(WritableFont.ARIAL, 10, WritableFont.NO_BOLD, false,
UnderlineStyle.NO_UNDERLINE, jxl.format.Colour.RED);
jxl.write.WritableCellFormat wcfFC = new jxl.write.WritableCellFormat(wfc);
jxl.write.Label labelCFC = new jxl.write.Label(1, 0, "This is a Label Cell", wcfFC);
ws.addCell(labelCF);
//2.添加Number对象
jxl.write.Number labelN = new jxl.write.Number(0, 1, 3.1415926);
ws.addCell(labelN);
//添加带有formatting的Number对象
jxl.write.NumberFormat nf = new jxl.write.NumberFormat("#.##");
jxl.write.WritableCellFormat wcfN = new jxl.write.WritableCellFormat(nf);
jxl.write.Number labelNF = new jxl.write.Number(1, 1, 3.1415926, wcfN);
ws.addCell(labelNF);
//3.添加Boolean对象
jxl.write.Boolean labelB = new jxl.write.Boolean(0, 2, false);
ws.addCell(labelB);
//4.添加DateTime对象
jxl.write.DateTime labelDT = new jxl.write.DateTime(0, 3, new java.util.Date());
ws.addCell(labelDT);
//添加带有formatting的DateFormat对象
jxl.write.DateFormat df = new jxl.write.DateFormat("dd MM yyyy hh:mm:ss");
jxl.write.WritableCellFormat wcfDF = new jxl.write.WritableCellFormat(df);
jxl.write.DateTime labelDTF = new jxl.write.DateTime(1, 3, new java.util.Date(), wcfDF);
ws.addCell(labelDTF);

4.2   修改excel文件代码举例

/**
     * <p>Description: 初始化模板信息</p>
     * <p>Remark: </p>
             * @return
     * @throws ServiceException
     */
    public AccessoryVO doInit(String initExtId,String departmentCode) throws ServiceException
    {
        AccessoryVO accessoryVO = new AccessoryVO();
        /*重新初始化,通过初始化编号获得规划模板编号*/
        String templateExtId = initDAO.getTemplateIdByInitId(initExtId);
        String rpSno = initDAO.getRpSno(templateExtId);
        FilloutVO filloutVO = new FilloutVO();
        filloutVO.setRpSno(rpSno);
        filloutVO.setDepartmentCode(departmentCode);
        List list= initDAO.getHistoryData(filloutVO);
       // initDAO.updateFilloutStatus("5",initExtId);
        /*获得初始化模板*/
        InputStream ins = initDAO.getExcelTemplate(templateExtId);
        /*获得空附件输出流*/
        ByteArrayOutputStream outs = new ByteArrayOutputStream();
      
        createExcel(ins,outs,list);
        accessoryVO.setAccessoryBody(outs.toByteArray());
        //initDAO.updateRpStatus("5",rpSno);
        try
        {
            if(outs!=null)
            {
                outs.close();
            }
            if(ins!=null)
            {
                ins.close();
            }
        }
        catch (IOException e)
        {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
      
        return accessoryVO;
    }
 
    /**
     * <p>Description: 生成带初始化数据的excel</p>
     * <p>Remark: </p>
     * @return
     * @throws ServiceException
     */
    public String createExcel(InputStream ins,OutputStream outs,List list)
            throws ServiceException
    {
            changeWorkbookByStream(ins,outs);
             try
             {
             WritableWorkbook book = null;
             book = this.wwbook;
             //设置格式
             jxl.write.NumberFormat nf = new jxl.write.NumberFormat("#,###");
             jxl.write.WritableCellFormat wcfN = new jxl.write.WritableCellFormat(nf);
             wcfN.setBorder(Border.ALL,BorderLineStyle.THIN);
             wcfN.setVerticalAlignment(VerticalAlignment.CENTRE);
           
             int length = list.size();
             WritableSheet rs = null;
             RdVO vo = null;
             //循环取得处理sheet数据
             for(int j = 0;j<length;j++){
                 vo = (RdVO)list.get(j);
                 sheetId = vo.getSheetId();
                 cellRow = vo.getCellRow();
                 cellColumn = vo.getCellColumn();
                 rpData = NullProcessUtil.nvlToString(vo.getRpData(), "0");
                 rs = (WritableSheet)book.getSheet(Integer.parseInt(sheetId));
                 labeltemp = new Number( Integer.parseInt(cellColumn),Integer.parseInt(cellRow), Double.parseDouble(rpData),wcfN);
                 ((WritableSheet) rs).addCell(labeltemp);
             }
             book.write();
             //关闭book,释放资源
             book.close();
             rwbook.close();
             }catch(Exception e){      
                 errorMsg = e.getMessage();
                 //errorMsg += "  错误在行 :"+rowTemp+" 列:"+colTemp;
                 System.out.println("出错了啊!!"+errorMsg);
                 System.out.println(e);
                 e.printStackTrace();
                 return this.errorMsg;
             }
             return this.errorMsg;
        }
    /**
     * 变更取得Workbook的方式为流获取
     * @author
     * @date Jun 2, 2008
     * @return void
     */
    public void changeWorkbookByStream(InputStream instream,OutputStream outstream) {
                try {
                    this.rwbook = Workbook.getWorkbook(instream);
                    System.out.println("this.wbook init() OK!!");
                } catch (IOException e) {
                    e.printStackTrace();
                } catch (BiffException e) {
                    e.printStackTrace();
                }
                try {
                    this.wwbook = Workbook.createWorkbook(outstream, this.rwbook,new WorkbookSettings());
                   
                   
                } catch (IOException e) {
                    e.printStackTrace();
                }
               
    }