Java软件工程师从笨鸟到初学者之(一百零五)java操作office和pdf文件(三)利用jxl实现数据导出excel报表以及与POI的区别

Java程序员从笨鸟到菜鸟之(一百零五)java操作office和pdf文件(三)利用jxl实现数据导出excel报表以及与POI的区别


        在上一篇博客中我们介绍了利用Apache开源项目POI实现数据库数据导出excel报表的知识。其实Java世界里,有两套成熟的开源工具支持对Excel文件的操作:一个是Apache POI;另一个则是本文将要介绍的Java Excel APIApache POIApache软件基金会的开放源码函式库,POI提供APIJava程式对Microsoft Office格式档案读/写功能。该项目分为几个组件,其中包括一个HSSF的组件,它是一个非常正规和严谨的API。利用HSSF,你可以用纯Java代码来读取、写入、修改Excel文件 。Java Excel APIJXL),它是一套纯粹使用Java开发的Excel表格操作组件,使用它,即使在非Windows操作系统下,程序员也可以通过纯 Java 应用来创建Excel文件,并能够读取,写入,更新Excel中的内容,因此,基于JXL可以实现对数据库中数据导入导出的操作。与POI相比,JXL拥有更小的内存占用率和对中文更好的支持,所以这篇文字我们就来看一下数据库数据导入导出基于JXL的方法。顺便介绍一下jxlPOI的区别。其实从原理上,个人感觉这两种方式很相似,就连代码也挺像。下面我们就来看一下简单的操作步骤吧。


jxl下载地址http://www.andykhan.com/jexcelapi/download.html 

基本步骤:


1.创建一个工作薄

WritableWorkbook book=Workbook.createWorkbook(new File(测试.xls)); 


2.生成名为第一页的工作表,参数0表示这是第一页  

WritableSheet sheet=book.createSheet(第一页,0); 


3.创建单元格Label对象的构造方法中指明单元格位置是第一列第一行(0,0)以及单元格内容为test 

Label label=new Label(0,0,test); 


4.将定义好的单元格添加到工作表中  

sheet.addCell(label); 

Jxl提供了一个专门创造数字类型的单元格,生成一个保存数字的单元格必须使用Number的完整包路径,否则有语法歧义单元格位置是第二列,第一行,值为123.123*/

jxl.write.Number number=new jxl.write.Number(1,0,123.123);sheet.addCell(number); 


5.写入数据并关闭文件  

book.write();

book.close(); 

关于Excle的一些操作:

1、 字串格式化

 字符串的格式化涉及到的是字体、粗细、字号等元素,这些功能主要由WritableFont
 WritableCellFormat类来负责。假设我们在生成一个含有字串的单元格时,使用如下语句,
 为方便叙述,我们为每一行命令加了编号:


WritableFont font1 = 
  new  WritableFont(WritableFont.TIMES, 16 ,WritableFont.BOLD); ①

 WritableCellFormat format1 = new  WritableCellFormat(font1); ②

 Label label = new  Label( 0 , 0 ,”data  4  test”,format1) ③ 


 其中指定了字串格式:字体为TIMES,字号16,加粗显示。WritableFont有非常丰富的
 构造子,供不同情况下使用,jExcelAPIjava-doc中有详细列表,这里不再列出。

 ②处代码使用了WritableCellFormat类,这个类非常重要,通过它可以指定单元格的各种
 属性,后面的单元格格式化中会有更多描述。

 ③处使用了Label类的构造子,指定了字串被赋予那种格式。

WritableCellFormat类中,还有一个很重要的方法是指定数据的对齐方式,比如针对我们
 上面的实例,可以指定:

   // 把水平对齐方式指定为居中 
  format1.setAlignment(jxl.format.Alignment.CENTRE);

  // 把垂直对齐方式指定为居中 
  format1.setVerticalAlignment(jxl.format.VerticalAlignment.CENTRE);


2、单元格操作

 Excel中很重要的一部分是对单元格的操作,比如行高、列宽、单元格合并等,所幸jExcelAPI
 提供了这些支持。这些操作相对比较简单,下面只介绍一下相关的API


 1)、 合并单元格

  WritableSheet.mergeCells( int  m, int  n, int  p, int  q); 

  // 作用是从(m,n)(p,q)的单元格全部合并,比如: 
  WritableSheet sheet = book.createSheet(第一页, 0 );

  // 合并第一列第一行到第六列第一行的所有单元格 
  sheet.mergeCells( 0 , 0 , 5 , 0 );

 合并既可以是横向的,也可以是纵向的。合并后的单元格不能再次进行合并,否则会触发异常。


 2)、 行高和列宽

  WritableSheet.setRowView( int  i, int  height);

  // 作用是指定第i+1行的高度,比如:

  // 将第一行的高度设为200 
  sheet.setRowView( 0 , 200 );

 WritableSheet.setColumnView( int  i, int  width);

  // 作用是指定第i+1列的宽度,比如:

  // 将第一列的宽度设为30 
  sheet.setColumnView( 0 , 30 );

 

jxl还有其他的一些功能,比如插入图片等,这里就不再一一介绍,读者可以自己探索。


POIjxl区别:


JVM虚拟机内存消耗的情况.

数据量3000条数据,每条60.JVM虚拟机内存大小64M.

使用POI:运行到2800条左右就报内存溢出.

使用JXL:3000条全部出来,并且内存还有21M的空间.

可想而知,在对内存的消耗方面差距还是挺大的.

也许是由于JXL在对资源回收利用方面做的还挺不错的.


提供的功能的情况

JXL相对弱了点.没有提供像POI那样复杂的功能,比如添加图片功能。所以如果要实现的功能比较复杂的情况下可以考虑使用POI


最后来看一个小例子:


import java.io.File;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import jxl.Workbook;
import jxl.write.Label;
import jxl.write.WritableCellFormat;
import jxl.write.WritableFont;
import jxl.write.WritableSheet;
import jxl.write.WritableWorkbook;
public class DateToExcel {
 private String driverClass = "com.mysql.jdbc.Driver";
 private String url = "jdbc:MySQL://localhost/boutiquecourse";
 private String user = "root";
 private String password = "";
 private Connection connection;
 public void exportClassroom(OutputStream os) {
  try {
   WritableWorkbook wbook = Workbook.createWorkbook(os); // 建立excel文件
   WritableSheet wsheet = wbook.createSheet("测试转换", 0); // 工作表名称
   //设置Excel字体
   WritableFont wfont = new WritableFont(WritableFont.ARIAL, 16,
    WritableFont.BOLD, false,
   jxl.format.UnderlineStyle.NO_UNDERLINE,
   jxl.format.Colour.BLACK);
   WritableCellFormat titleFormat = new WritableCellFormat(wfont);
   
   String[] title = { "测试编号", "测试名称"};//如果还有字段的话,以此类推
   // 设置Excel表头
   for (int i = 0; i < title.length; i++) {
    Label excelTitle = new Label(i, 0, title[i], titleFormat);
    wsheet.addCell(excelTitle);
   }
   int c = 1; // 用于循环时Excel的行号
   Connection con = openConnection();
   Statement st = con.createStatement();
   String sql = "select * from test";
   ResultSet rs = st.executeQuery(sql); // 这个是从数据库中取得要导出的数据
   while (rs.next()) {
    Label content1 = new Label(0, c, (String) rs.getString("testid"));
    Label content2 = new Label(1, c, (String) rs.getString("testname"));
    //如果还有的话,以此类推
    wsheet.addCell(content1);
    wsheet.addCell(content2);
    //如果还有的话,以此类推
    c++;
   }
   wbook.write(); // 写入文件
   wbook.close();
   os.close();
   System.out.println("导入成功!");
  } catch (Exception e) {
   e.printStackTrace();
  }
 }
 public Connection openConnection() throws SQLException {
  try {
   Class.forName(driverClass).newInstance();
   connection = DriverManager.getConnection(url, user, password);
   return connection;
  } catch (Exception e) {
   throw new SQLException(e.getMessage());
  }
 }
 public void closeConnection() {
  try {
   if (connection != null)
    connection.close();
  } catch (Exception e) {
   e.printStackTrace();
  }
 }
 public static void main(String[] args) {
  DateToExcel te = new DateToExcel();
  File f = new File("D:/kk.xls");
  //File f = new File("D:\\kk.xls");
  try {
   f.createNewFile();
   OutputStream os = new FileOutputStream(f);
   te.exportClassroom(os);
  } catch (Exception e) {
   e.printStackTrace();
  }
 }
}


参考资料:百度百科:jxl.jar

------------------------------------------------

  广告:我参加了2012年度IT博客大赛,希望大家能多多支持

    http://blog.51cto.com/contest2012/3545281

------------------------------------------------------------------------

《Java程序员由笨鸟到菜鸟》电子版书正式发布,欢迎大家下载


http://blog.csdn.net/csh624366188/article/details/7999247




3楼zcj11131小时前
2楼bzu9021小时前
不错。。。加油。
1楼csh6243661881小时前
虽然简单,但却真有用处。。