在我们实际的开发中,表现层的解决方案虽然有多样,但是ie浏览器已成为最多人使用的浏览器,因为大家都用windows。在企业办公系统中,常常有客户这样子要求:你要把我们的报表直接用excel打开。或者是:我们已经习惯用excel打印。这样子如果用.net开发是没有问题的,.net用来进行这些数据导入导出支持的都是很好,如果你使用的是sql server数据库那就更方便了,笔者也曾体会到.net开发这些功能的方便性.但是有j2ee这个比.net更有前途的开放式的开发环境,难道我为了解决打印的要求去另写客户端的控件?或者在服务器端使用本地代码?第一种方案的问题是关键数据的处理有时候不能在客户端做,第2种方案的问题是牺牲了代码的可移植性和稳定性。如果让客户端只负责处理生成好的报表,那将是一种诱人的选择。
apache的jakata项目的poi子项目,目标是处理ole2对象。目前比较成熟的是hssf接口,处理ms excel(97-2002)对象。它不象我们仅仅是用csv生成的没有格式的可以由excel转换的东西,而是真正的excel对象,你可以控制一些属性如sheet,cell等等。这是一个年轻的项目,所以象hdf这样直接支持word对象的好东西仍然在设计中。其它支持word格式的纯java方案还有itext,不过也是仍在奋斗中。但是hssf已经成熟到能够和足够我们使用了。另外,无锡永中office的实现方案也是纯java的解决方案,不过那也是完全商业的产品,并不是公开代码项目。其实,从开发历史的角度讲,在80年代中期staroffice的原作者在德国成立了staroffice suite公司,然后到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的处理上很烦恼。还好在poi hssf中考虑到这个问题,可以设置encoding为双字节。
poi可以到www.apache.org下载到。编译好的jar主要有这样4个:poi包,poi browser包,poi hdf包,poi hssf例程包。实际运行时,需要有poi包就可以了。如果用jakarta ant编译和运行,下载apache jakarta poi的release中的src包,它里面已经为你生成好了build文件了。只要运行ant就可以了(ant 的安装和使用在此不说了)。如果是用jbuilder 运行,请在新建的项目中加入poi包。以jbuilder6为例,选择tools菜单项的config libraries...选项,新建一个lib。在弹出的菜单中选择poi包,如这个jakarta-poi-1.5.1-final-20020820.jar,把poi添加到jbuilder中。然后,右键点击你的项目,在project的properties菜单中path的required libraries中,点add,添加刚才加入到jbuilder中的poi到你现在的项目中。如果你仅仅是为了熟悉poi hssf的使用,可以直接看poi的samples包中的源代码,并且运行它。hssf的各种对象都有例程的介绍。hssf提供的例程在org.apache.poi.hssf.usermodel.examples包中,共有14个,生成的目标xls都是workbook.xls。如果你想看更多的例程,可以参考hssf的junit test cases,在poi的包的源代码中有。hssf都有测试代码。
首先,到http://apache.justdn.org/jakarta/poi/release/bin/上去下载poi,登录到该网址以后,选择红色区域的poi-bin-2.5.1-final-20040804.zip来下载
下载以后解压,解压目录为
在eclipse中设置对poi-bin-2.5.1-final-20040804.jar的库引用.
在eclipse中新建一个名字叫poiexcel的java工程,右键poiexcel项目的项目名,选择build path,再选择add external archives,然后找到poi-bin-2.5.1-final-20040804.jar这个包将其引进到工程中来.
excel 结构
hssfworkbook excell 文档对象介绍
hssfsheet excell的表单
hssfrow excell的行
hssfcell excell的格子单元
hssffont excell字体
hssfname 名称
hssfdataformat 日期格式
辅助操作包括
hssfdateutil 日期
hssfprintsetup 打印
hssferrorconstants 错误信息表
将数据导出到excel中的实例
1. 创建一个空白的excel文件
我们创建一个最普通的java应用程序即可,程序代码如下: hssfworkbook wb=new hssfworkbook();
fileoutputstream fileout = new fileoutputstream(c://test.xls);
wb.write(fileout);
fileout.close();
fileoutputstream属于jdk的数据流对象,通过它来将一个名为test.xls文件创建在c盘根目录下,运行该程序后可以看到在c盘上的根目录上多了一个excel文件.
如果将程序中的fileoutputstream(c://test.xls)的参数改为test.xls,则文件就会创建在该项目的根目录下.
2. 往excel的单元格中写入信息
poi把excel分解成几个对象,自顶向下的关系是:工作薄(hssfworkbook)->表格(hssfsheet)->表格行(hssfrow)->单元格(hssfcell).将数据库中表的数据导入到excel中,首先我们要将数据库中的表的数据查询出来,这里我们通过hibernate来将数据库中addressbook_table表的数据查询出来.然后通过循环将表中的数据导入到excel中去. hssfworkbook wb=new hssfworkbook();
hssfsheet sheet=wb.createsheet(new sheet);
try ...{
//通过hibernate来查询addressbook_table表中的数据,将其存储在list中
transaction tx = session.begintransaction();
org.hibernate.query query= session.createquery(from addressbooktable);
list list = query.list();
tx.commit();
int k =0;
//创建表格,创建表格行和单元格,将数据库中表的字段存储在单元格中.
for(iterator it=list.iterator();it.hasnext();)...{
addressbooktable user =(addressbooktable)it.next();
hssfrow row=sheet.createrow((short)k);
row.createcell((short)0).setcellvalue(user.getid());
row.createcell((short)1).setcellvalue(user.getname());
row.createcell((short)2).setcellvalue(user.getaddress());
row.createcell((short)3).setcellvalue(user.getphone());
k++;
}
fileoutputstream fileout = new fileoutputstream(c://test.xls);
wb.write(fileout);
fileout.close();
log.error(successful!!);
} catch (hibernateexception e) ...{
log.error(insert exception!);
e.printstacktrace();
}