java操作ExcelJakartaPOI.docx
《java操作ExcelJakartaPOI.docx》由会员分享,可在线阅读,更多相关《java操作ExcelJakartaPOI.docx(23页珍藏版)》请在冰点文库上搜索。
![java操作ExcelJakartaPOI.docx](https://file1.bingdoc.com/fileroot1/2023-5/28/77003455-f03a-44c8-afd5-4c524f7a4262/77003455-f03a-44c8-afd5-4c524f7a42621.gif)
java操作ExcelJakartaPOI
java操作Excel(Jakarta_POI)
今天又用到了excel操作但由于相隔久远只好重新复习,顺便整理一下,之前整理了JavaExcelAPI(简称JXL)的,由于POI用的不是很多就把4篇整理到一个blog中!
!
akataPoiHSSF:
纯java的Excel解决方案
2002-10-23上午08:
52:
34
微软在桌面系统上的成功,令我们不得不大量使用它的办公产品,如:
Word,Excel。
时至今日,它的源代码仍然不公开已封锁了我们的进一步应用和开发。
然而在要求更高的服务器领域,微软本身的产品移植性不好,
性能不佳。
在我们实际的开发中,表现层的解决方案虽然有多样,但是Ie浏览器已成为最多人使用的浏览器,因为大家都用Windows。
在企业办公系统中,常常有客户这样子要求:
你要把我们的报表直接用Excel打开。
或者是:
我们已经习惯用Excel打印。
这样子如果用.net开发是没有问题的,但是有j2ee这个比.net更有前途的开放式的开发环境,难道我为了解决打印的要求去另写客户端的控件?
或者在服务器端使用本地代码?
第一种方案的问题是关键数据的处理有时候不能在客户端做,第2种方案的问题是牺牲了代码的可移植性和稳定性。
如果让客户端只负责处理生成好的报表,那将是一种诱人的选择。
Apache的Jakata项目的POI子项目,目标是处理ole2对象。
目前比较成熟的是HSSF接口,处理MSExcel(97-2002)对象。
它不象我们仅仅是用csv生成的没有格式的可以由Excel转换的东西,而是真正的Excel对象,你可以控制一些属性如sheet,cell等等。
这是一个年轻的项目,所以象HDF这样直接支持Word对象的好东西仍然在设计中。
其它支持word格式的纯java方案还有itext,不过也是仍在奋斗中。
但是HSSF已经成熟到能够和足够我们使用了。
另外,无锡永中Office的实现方案也是纯java的解决方案,不过那也是完全商业的产品,并不是公开代码项目。
其实,从开发历史的角度讲,在80年代中期starOffice的原作者在德国成立了StarOfficesuite公司,然后到1999年夏天starOffice被sun收购,再到2000年6月starOffice5.2的发布;并且从starOffice6.0开始,starOffice建立在OpenOffice的api的基础上,这个公开代码的office项目已经进行了很长的时间。
虽然那是由C++写的,但是POI的代码部分也是由openOffice改过来的。
所以,应该对POI充满足够的信心。
国内已经有部分公司在他们的办公自动化等Web项目中使用poi了,如日恒的ioffice,海泰的HTOffice等。
java当初把核心处理设成Unicode,带来的好处是另代码适应了多语言环境。
然而由于老外的英语只有26个字母,有些情况下,一些程序员用8位的byte处理,一不小心就去掉了CJK的高位。
或者是由于习惯在程序中采用硬编码,还有多种原因,使得许多java应用在CJK的处理上很烦恼。
还好在POIHSSF中考虑到这个问题,可以设置encoding为双字节。
POI可以到www.apache.org下载到。
编译好的jar主要有这样4个:
poi包,poiBrowser包,poihdf包,poihssf例程包。
实际运行时,需要有poi包就可以了。
如果用Jakartaant编译和运行,下载apacheJakartaPOI的release中的src包,它里面已经为你生成好了build文件了。
只要运行ant就可以了(ant的安装和使用在此不说了)。
如果是用Jbuilder运行,请在新建的项目中加入poi包。
以Jbuilder6为例,选择Tools菜单项的configlibraries...选项,新建一个lib。
在弹出的菜单中选择poi包,如这个jakarta-poi-1.5.1-final-20020820.jar,把poi添加到jbuilder中。
然后,右键点击你的项目,在project的properties菜单中path的requiredLibraries中,点add,添加刚才加入到jbuilder中的poi到你现在的项目中。
如果你仅仅是为了熟悉POIhssf的使用,可以直接看POI的samples包中的源代码,并且运行它。
hssf的各种对象都有例程的介绍。
hssf提供的例程在org.apache.poi.hssf.usermodel.examples包中,共有14个,生成的目标xls都是workbook.xls。
如果你想看更多的例程,可以参考hssf的Junittestcases,在poi的包的源代码中有。
hssf都有测试代码。
这里只对部分例程的实现做介绍。
HSSF提供给用户使用的对象在org.apache.poi.hssf.usermodel包中,主要部分包括Excell对象,样式和格式,还有辅助操作。
有以下几种对象:
HSSFWorkbookexcell的文档对象
HSSFSheetexcell的表单
HSSFRowexcell的行
HSSFCellexcell的格子单元
HSSFFontexcell字体
HSSFName名称
HSSFDataFormat日期格式
在poi1.7中才有以下2项:
HSSFHeadersheet头
HSSFFootersheet尾
和这个样式
HSSFCellStylecell样式
辅助操作包括
HSSFDateUtil日期
HSSFPrintSetup打印
HSSFErrorConstants错误信息表
仔细看org.apache.poi.hssf包的结构,不难发现HSSF的内部实现遵循的是MVC模型。
这里我用Rose把org.apache.poi.hssf.usermodel包中的对象反向导入并根据相互关系作了整理,详见下面两图:
图1基本对象
从中不难可以发现每一个基本对象都关联了一个Record对象。
Record对象是一个参考Office格式的相关记录。
图2HSSFWorkbook
HSSFWorkbook即是一个Excell对象。
这幅类图体现的是HSSFWorkbook和基本对象的相互关系。
可见,许多对象中也建立了Workbook的引用。
还需要注意的是在HSSFWorkbook和HSSFSheet中建立了log机制POILogger,而且POILogger也是使用apacheLog4J实现的。
先看poi的examples包中提供的最简单的例子,建立一个空xls文件。
importorg.apache.poi.hssf.usermodel.HSSFWorkbook;
importjava.io.FileOutputStream;
importjava.io.IOException;
publicclassNewWorkbook
{
publicstaticvoidmain(String[]args)
throwsIOException
{
HSSFWorkbookwb=newHSSFWorkbook();//建立新HSSFWorkbook对象
FileOutputStreamfileOut=newFileOutputStream("workbook.xls");
wb.write(fileOut);//把Workbook对象输出到文件workbook.xls中
fileOut.close();
}
}
通过这个例子,我们建立的是一个空白的xls文件(不是空文件)。
在此基础上,我们可以进一步看其它的例子。
importorg.apache.poi.hssf.usermodel.*;
importjava.io.FileOutputStream;
importjava.io.IOException;
publicclassCreateCells
{
publicstaticvoidmain(String[]args)
throwsIOException
{
HSSFWorkbookwb=newHSSFWorkbook();//建立新HSSFWorkbook对象
HSSFSheetsheet=wb.createSheet("newsheet");//建立新的sheet对象
//Createarowandputsomecellsinit.Rowsare0based.
HSSFRowrow=sheet.createRow((short)0);//建立新行
//Createacellandputavalueinit.
HSSFCellcell=row.createCell((short)0);//建立新cell
cell.setCellValue
(1);//设置cell的整数类型的值
//Ordoitononeline.
row.createCell((short)1).setCellValue(1.2);//设置cell浮点类型的值
row.createCell((short)2).setCellValue("test");//设置cell字符类型的值
row.createCell((short)3).setCellValue(true);//设置cell布尔类型的值
HSSFCellStylecellStyle=wb.createCellStyle();//建立新的cell样式
cellStyle.setDataFormat(HSSFDataFormat.getFormat("m/d/yyh:
mm"));//设置cell样式为定制的日期格式
HSSFCelldCell=row.createCell((short)4);
dCell.setCellValue(newDate());//设置cell为日期类型的值
dCell.setCellStyle(cellStyle);//设置该cell日期的显示格式
HSSFCellcsCell=row.createCell((short)5);
csCell.setEncoding(HSSFCell.ENCODING_UTF_16);//设置cell编码解决中文高位字节截断
csCell.setCellValue("中文测试_ChineseWordsTest");//设置中西文结合字符串
row.createCell((short)6).setCellType(HSSFCell.CELL_TYPE_ERROR);//建立错误cell
//Writetheoutputtoafile
FileOutputStreamfileOut=newFileOutputStream("workbook.xls");
wb.write(fileOut);
fileOut.close();
}
}
我稍微修改了原来的examples包中的CreateCells类写了上面的功能测试类。
通过这个例子,我们可以清楚的看到xls文件从大到小包括了HSSFWorkbookHSSFSheetHSSFRowHSSFCell这样几个对象。
我们可以在cell中设置各种类型的值。
尤其要注意的是如果你想正确的显示非欧美的字符时,尤其象中日韩这样的语言,必须设置编码为16位的即是HSSFCell.ENCODING_UTF_16,才能保证字符的高8位不被截断而引起编码失真形成乱码。
其他测试可以通过参考examples包中的测试例子掌握poi的详细用法,包括字体的设置,cell大小和低纹的设置等。
需要注意的是POI是一个仍然在完善中的公开代码的项目,所以有些功能正在不断的扩充。
如HSSFSheet的getFooter()getHeader()和setFooter(HSSFFooterhsf)setHeader(HSSFHeaderhsh)是在POI1.7中才有的,而POI1.5中就没有。
运行测试熟悉代码或者使用它做项目时请注意POI的版本。
另外需要注意的是HSSF也有它的对xls基于事件的解析。
可以参考例程中的EventExample.java。
它通过实现HSSFListener完成从普通流认知Xls中包含的内容,在apacheCocoon中的org.apache.cocoon.serialization.HSSFSerializer中用到了这个解析。
因为Cocoon2是基于事件的,所以POI为了提供快速的解析也提供了相应的事件。
当然我们自己也可以实现这个事件接口。
因为POI还不是一个足够成熟的项目,所以有必要做进一步的开发和测试。
但是它已经为我们用纯java操作ole2对象提供了可能,而且克服了ole对象调用的缺陷,提供了服务器端的Excel解决方案。
======================================================
利用Java创建和读取Excel文档
--------------------------------------------------------------------------------
源作者:
Rubber 人气:
26748
为了保证示例程序的运行,必须安装Java2sdk1.4.0和JakartaPOI,JakartaPOI的Web站点是:
http:
//jakarta.apache.org/poi/
示例1将演示如何利用JakartaPOIAPI创建Excel文档。
示例1程序如下:
importorg.apache.poi.hssf.usermodel.HSSFWorkbook;
importorg.apache.poi.hssf.usermodel.HSSFSheet;
importorg.apache.poi.hssf.usermodel.HSSFRow;
importorg.apache.poi.hssf.usermodel.HSSFCell;
importjava.io.FileOutputStream;
publicclassCreateXL{
/**Excel文件要存放的位置,假定在D盘JTest目录下*/
publicstaticStringoutputFile="D:
/JTest/gongye.xls";
publicstaticvoidmain(Stringargv[])
{
try
{
//创建新的Excel工作簿
HSSFWorkbookworkbook=newHSSFWorkbook();
//在Excel工作簿中建一工作表,其名为缺省值
//如要新建一名为"效益指标"的工作表,其语句为:
//HSSFSheetsheet=workbook.createSheet("效益指标");
HSSFSheetsheet=workbook.createSheet();
//在索引0的位置创建行(最顶端的行)
HSSFRowrow=sheet.createRow((short)0);
//在索引0的位置创建单元格(左上端)
HSSFCellcell=row.createCell((short)0);
//定义单元格为字符串类型
cell.setCellType(HSSFCell.CELL_TYPE_STRING);
//在单元格中输入一些内容
cell.setCellValue("增加值");
//新建一输出文件流
FileOutputStreamfOut=newFileOutputStream(outputFile);
//把相应的Excel工作簿存盘
workbook.write(fOut);
fOut.flush();
//操作结束,关闭文件
fOut.close();
System.out.println("文件生成...");
}catch(Exceptione){
System.out.println("已运行xlCreate():
"+e);
}
}
}
读取Excel文档中的数据
示例2将演示如何读取Excel文档中的数据。
假定在D盘JTest目录下有一个文件名为gongye.xls的Excel文件。
示例2程序如下:
importorg.apache.poi.hssf.usermodel.HSSFWorkbook;
importorg.apache.poi.hssf.usermodel.HSSFSheet;
importorg.apache.poi.hssf.usermodel.HSSFRow;
importorg.apache.poi.hssf.usermodel.HSSFCell;
importjava.io.FileInputStream;
publicclassReadXL{
/**Excel文件的存放位置。
注意是正斜线*/
publicstaticStringfileToBeRead="D:
/JTest/gongye.xls";
publicstaticvoidmain(Stringargv[]){
try{
//创建对Excel工作簿文件的引用
HSSFWorkbookworkbook=newHSSFWorkbook(newFileInputStream(fileToBeRead));
//创建对工作表的引用。
//本例是按名引用(让我们假定那张表有着缺省名"Sheet1")
HSSFSheetsheet=workbook.getSheet("Sheet1");
//也可用getSheetAt(intindex)按索引引用,
//在Excel文档中,第一张工作表的缺省索引是0,
//其语句为:
HSSFSheetsheet=workbook.getSheetAt(0);
//读取左上端单元
HSSFRowrow=sheet.getRow(0);
HSSFCellcell=row.getCell((short)0);
//输出单元内容,cell.getStringCellValue()就是取所在单元的值
System.out.println("左上端单元是:
"+cell.getStringCellValue());
}catch(Exceptione){
System.out.println("已运行xlRead():
"+e);
}
}
}
设置单元格格式
在这里,我们将只介绍一些和格式设置有关的语句,我们假定workbook就是对一个工作簿的引用。
在Java
中,第一步要做的就是创建和设置字体和单元格的格式,然后再应用这些格式:
1、创建字体,设置其为红色、粗体:
HSSFFontfont=workbook.createFont();
font.setColor(HSSFFont.COLOR_RED);
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
2、创建格式
HSSFCellStylecellStyle=workbook.createCellStyle();
cellStyle.setFont(font);
3、应用格式
HSSFCellcell=row.createCell((short)0);
cell.setCellStyle(cellStyle);
cell.setCellType(HSSFCell.CELL_TYPE_STRING);
cell.setCellValue("标题");
总之,如本篇文章所演示的一样,Java程序员不必担心Excel工作表中的数据了,利用JakartaPOIAPI,
我们就可以轻易的在程序中存取Excel文档。
==============================================
首先说说现在我所知道的Java编辑Excel文件的两大开源工具:
jakartaPOI和JavaExcelAPI(简称JXL),这两套工具我都试用了一这段时间,感觉各有优劣吧。
POI在某些细节有些小Bug并且不支持写入图片,其他方面都挺不错的;
JXL就惨了,除了支持写入图片外,我暂时看不到它比POI好的地方,我碰到的主要的问题就是对公式支持不是很好,很多带有公式的Excel文件用JXL打开后,公式就丢失了(比如now(),today()),在网上看到其他大虾评论说JXL写入公式也有问题,另外,JXL操作Excel文件的效率比POI低一点。
经过比较后,我选择了POI开发我的项目。
现在我要做的东西基本完成啦,我把这段时间使用POI的一些心得总结出来,希望能对和我遇到相同问题的朋友有所帮助,至于POI基本的使用方法,自己去看文档吧。
1、设置分页符的bug
POI里的HSSFSheet类提供了setRowBreak方法可以设置Sheet的分页符。
Bug:
如果你要设置分页符的Sheet是本来就有的,并且你没有在里面插入过分页符,那么调用setRowBreak时POI会抛出空指针的异常。
解决方法:
在Excel里给这个sheet插入一个分页符,用POI打开后再把它删掉,然后你就可以随意插入分页符了。
如果sheet是由POI生成的则没有这个问题。
我跟踪了setRowBreak的源代码,发现是Sheet.Java下的PageBreakRecordrowBreaks这个变量在搞鬼,如果Sheet里原来没有分页符,开发这个模块的那位兄台忘了为这个对象new实例,所以只能我们先手工给Excel插入一个分页符来触发POI为rowBreaks创建实例。
2、如何拷贝行
我在gmane.org的POI用户论坛翻遍了每个相关的帖子,找遍了api,也没看到一个拷贝行的方法,没办法,只能自己写:
//注:
this.fWorkbook是一个HSSHWorkbook,请自行在外部new
publicvoidcopyRows
(StringpSourceSheetName,
StringpTargetSheetName,
intpStartRow,intpEndRow,
intpPosition)
{
HSSFRowsourceRow=null;
HSSFRowtargetRow=null;
HSSFCellsourceCell=null;
HSSFCelltargetCell=null;
HSSFSheetsourceSheet=null;
HSSFSheettargetSheet=null;
Reg