JAVA EXCEL API:是一開放源碼項目,通過它Java開發人員可以讀取Excel文件的內容、創建新的Excel文件、更新已經存在的Excel文件。使用該API非Windows操作系統也可以通過純Java應用來處理Excel數據表。因爲它是使用Java編寫的,所以我們在Web應用中可以通過JSP、Servlet來調用API實現對Excel數據表的訪問。
Java Excel API的jar包可以通過以下URL獲得:
http://sourceforge.net/projects/jexcelapi/files/jexcelapi/2.6.6/jexcelapi_2_6_6.zip/download
(包括所有版本):http://sourceforge.net/projects/jexcelapi/files/
直接下載地址(迅雷上新建任務即可):
http://nchc.dl.sourceforge.net/project/jexcelapi/jexcelapi/2.6.6/jexcelapi_2_6_6.zip
示例1:讀取本地Excel文件F:\紅樓人物.xls
1. 新建Excel文件F:\紅樓人物.xls
內容如下:
2. Java通過jexcelapi包操作excel文件:
//in ExcelOperater
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import jxl.Cell;
import jxl.CellType;
import jxl.Sheet;
import jxl.Workbook;
import jxl.write.Label;
public class ExcelOperater
{
public static void main(String[] args)
{
jxl.Workbook readwb = null;
try
{
//構建Workbook對象, 只讀Workbook對象
//直接從本地文件創建Workbook
InputStream instream = new FileInputStream("F:/紅樓人物.xls");
readwb = Workbook.getWorkbook(instream);
//Sheet的下標是從0開始
//獲取第一張Sheet表
Sheet readsheet = readwb.getSheet(0);
//獲取Sheet表中所包含的總列數
int rsColumns = readsheet.getColumns();
//獲取Sheet表中所包含的總行數
int rsRows = readsheet.getRows();
//獲取指定單元格的對象引用
for (int i = 0; i < rsRows; i++)
{
for (int j = 0; j < rsColumns; j++)
{
Cell cell = readsheet.getCell(j, i);
System.out.print(cell.getContents() + " ");
}
System.out.println();
}
//利用已經創建的Excel工作薄,創建新的可寫入的Excel工作薄
jxl.write.WritableWorkbook wwb = Workbook.createWorkbook(new File(
"F:/紅樓人物1.xls"), readwb);
//讀取第一張工作表
jxl.write.WritableSheet ws =wwb.getSheet(0);
//獲得第一個單元格對象
jxl.write.WritableCell wc =ws.getWritableCell(0, 0);
//判斷單元格的類型, 做出相應的轉化
if (wc.getType() == CellType.LABEL)
{
Label l = (Label) wc;
l.setString("新姓名");
}
//寫入Excel對象
wwb.write();
wwb.close();
} catch (Exception e) {
e.printStackTrace();
} finally {
readwb.close();
}
}
}
3. 結果:
① 控制檯輸出:
人物 等級 大觀園位置 金陵十二釵
林黛玉 小姐 瀟湘館 正冊
妙玉 世外 櫳翠庵 正冊
晴雯 丫鬟 怡紅院 副冊
香菱 妾 蘅蕪苑 又副冊
② 創建文件F:\紅樓人物1.xls
4. 程序解析:
所引用的包:
① Workbook對象,需要jxl.Workbook包;
② InputStream、FileInputStream對象:需要java.io.FileInputStream和java.io.InputStream包。
③ Sheet對象:jxl.Sheet包;注意excel中sheet表單的行列從0開始計數。
④ Cell對象:jxl.Cell包;對單元進行處理
⑤ Label:選擇jxl.write.label包
⑥ WritableWorkbook、WritableSheet、WritableCell對象
實例二:3個功能-----從excel文件F:\紅樓人物.xls讀取數據;生成新的excel文件F:\紅樓人物2.xls;修改原excel一個單元並輸出爲F:\紅樓人物3.xls。
原始文件:F:\紅樓人物.xls
運行結果:
① 控制檯輸出:
人物 等級 大觀園位置 金陵十二釵
林黛玉 小姐 瀟湘館 正冊
妙玉 世外 櫳翠庵 正冊
晴雯 丫鬟 怡紅院 副冊
香菱 妾 蘅蕪苑 又副冊
② 寫入輸出Excel文件:F:\紅樓人物2.xls
③ 修改輸出文件 F:\紅樓人物3.xls (加修飾後輸出)
示例程序:
//in ExcelHandle
import jxl.*;
import jxl.format.UnderlineStyle;
import jxl.write.*;
import jxl.write.Number;
import jxl.write.Boolean;
import jxl.Cell;
import java.io.*;
public class ExcelHandle
{
public ExcelHandle()
{
}
/***讀取Excel*/
public static void readExcel(String filePath)
{
try
{
InputStream is = new FileInputStream(filePath);
Workbook rwb = Workbook.getWorkbook(is);
//這裏有兩種方法獲取sheet表:名字和下標(從0開始)
//Sheet st =rwb.getSheet("original");
Sheet st = rwb.getSheet(0);
/**
//獲得第一行第一列單元的值
Cell c00 = st.getCell(0,0);
//通用的獲取cell值的方式,返回字符串
String strc00 = c00.getContents();
//獲得cell具體類型值的方式
if(c00.getType() == CellType.LABEL)
{
LabelCell labelc00 = (LabelCell)c00;
strc00 = labelc00.getString();
}
//輸出
System.out.println(strc00);*/
//Sheet的下標是從0開始
//獲取第一張Sheet表
Sheet rst = rwb.getSheet(0);
//獲取Sheet表中所包含的總列數
int rsColumns = rst.getColumns();
//獲取Sheet表中所包含的總行數
int rsRows = rst.getRows();
//獲取指定單元格的對象引用
for (int i = 0; i < rsRows; i++)
{
for (int j = 0; j < rsColumns; j++)
{
Cell cell = rst.getCell(j, i);
System.out.print(cell.getContents() + " ");
}
System.out.println();
}
//關閉
rwb.close();
}
catch(Exception e)
{
e.printStackTrace();
}
}
/**輸出Excel*/
public static void writeExcel(OutputStream os)
{
try
{
/** 只能通過API提供的工廠方法來創建Workbook,而不能使用WritableWorkbook的構造函數,因爲類WritableWorkbook的構造函數爲protected類型:方法一:直接從目標文件中讀取WritableWorkbook wwb =Workbook.createWorkbook(new File(targetfile));方法二:如下實例所示 將WritableWorkbook直接寫入到輸出流*/
WritableWorkbook wwb = Workbook.createWorkbook(os);
//創建Excel工作表 指定名稱和位置
WritableSheet ws = wwb.createSheet("Test Sheet 1",0);
/**************往工作表中添加數據*****************/
//1.添加Label對象
Label label = new Label(0,0,"測試");
ws.addCell(label);
//添加帶有字型Formatting對象
WritableFont wf = newWritableFont(WritableFont.TIMES,18,WritableFont.BOLD,true);
WritableCellFormat wcf = new WritableCellFormat(wf);
Label labelcf = new Label(1,0,"this is a label test",wcf);
ws.addCell(labelcf);
//添加帶有字體顏色的Formatting對象
WritableFont wfc = newWritableFont(WritableFont.ARIAL,10,WritableFont.NO_BOLD,false,
UnderlineStyle.NO_UNDERLINE,jxl.format.Colour.DARK_YELLOW);
WritableCellFormatwcfFC = new WritableCellFormat(wfc);
Label labelCF = new Label(1,0,"Ok",wcfFC);
ws.addCell(labelCF);
//2.添加Number對象
Number labelN = new Number(0,1,3.1415926);
ws.addCell(labelN);
//添加帶有formatting的Number對象
NumberFormat nf = new NumberFormat("#.##");
WritableCellFormat wcfN = new WritableCellFormat(nf);
Number labelNF = new jxl.write.Number(1,1,3.1415926,wcfN);
ws.addCell(labelNF);
//3.添加Boolean對象
Boolean labelB = new jxl.write.Boolean(0,2,true);
ws.addCell(labelB);
Boolean labelB1 = new jxl.write.Boolean(1,2,false);
ws.addCell(labelB1);
//4.添加DateTime對象
jxl.write.DateTime labelDT = new jxl.write.DateTime(0,3,newjava.util.Date());
ws.addCell(labelDT);
//5.添加帶有formatting的DateFormat對象
DateFormat df = new DateFormat("dd MM yyyy hh:mm:ss");
WritableCellFormat wcfDF = new WritableCellFormat(df);
DateTime labelDTF = new DateTime(1,3,new java.util.Date(),wcfDF);
ws.addCell(labelDTF);
//6.添加圖片對象,jxl只支持png格式圖片
File image = new File("f:\\1.png");
WritableImage wimage = new WritableImage(0,4,6,17,image);
ws.addImage(wimage);
//7.寫入工作表
wwb.write();
wwb.close();
}
catch(Exception e)
{
e.printStackTrace();
}
}
/** 將file1拷貝後,進行修改並創建輸出對象file2
* 單元格原有的格式化修飾不能去掉,但仍可將新的單元格修飾加上去,
* 以使單元格的內容以不同的形式表現
*/
public static void modifyExcel(File file1,File file2)
{
try
{
Workbook rwb =Workbook.getWorkbook(file1);
WritableWorkbook wwb = Workbook.createWorkbook(file2,rwb);//copy
WritableFont wfc = newWritableFont(WritableFont.ARIAL,10,WritableFont.NO_BOLD,false,
UnderlineStyle.NO_UNDERLINE,jxl.format.Colour.BLUE);
WritableCellFormat wcfFC = new WritableCellFormat(wfc);
WritableSheet ws = wwb.getSheet(0);
WritableCell wc = ws.getWritableCell(0,0);
//判斷單元格的類型,做出相應的轉換
if(wc.getType() == CellType.LABEL)
{
Label labelCF =new Label(0,0,"人物(新)",wcfFC);
ws.addCell(labelCF);
//Label label = (Label)wc;
//label.setString("被修改");
}
wwb.write();
wwb.close();
rwb.close();
}
catch(Exception e)
{
e.printStackTrace();
}
}
//測試
public static void main(String args[])
{
try
{
//讀EXCEL
ExcelHandle.readExcel("F:/紅樓人物.xls");
//輸出EXCEL
File filewrite=new File("F:/紅樓人物2.xls");
filewrite.createNewFile();
OutputStream os=new FileOutputStream(filewrite);
ExcelHandle.writeExcel(os);
//修改EXCEL
ExcelHandle.modifyExcel(new File("F:/紅樓人物.xls"), new File("F:/紅樓人物3.xls"));
}
catch(Exception e)
{
e.printStackTrace();
}
}
}
附:
調用流程如下:
1.打開工作文件Workbook,在此之前先用java的io流創建或者讀取文件
2.打開工作表Sheet
3.讀行,然後讀列。注意,行和列是從零開始的
4.取得數據進行操作
來自網絡à讀取Excel數據表
第一步:創建Workbook(術語:工作薄)
2種方法:
//從輸入流創建Workbook讀取excel數據表
InputStream is = new FileInputStream(sourcefile);
jxl.Workbook workbook =Workbook.getWorkbook(is);
//直接從本地文件(.xls)創建Workbook
Workbook workbook = Workbook.getWorkbook(new File(excelfile));
一旦創建了Workbook,就可以通過它來訪問Excel Sheet(術語:工作表):
第二步:訪問sheet。
2種方法:通過sheet的名稱;或者通過下標,下標從0開始。
//獲取第一張Sheet表
Sheet rs = workbook.getSheet(0);
一旦得到了Sheet,就可以通過它來訪問Excel Cell(術語:單元格)。
第三步:訪問單元格cell
//獲取第一行,第一列的值
Cell c00 = rs.getCell(0, 0);
String strc00 = c00.getContents();
//獲取第一行,第二列的值
Cell c10 = rs.getCell(1, 0);
String strc10 = c10.getContents();
//獲取第二行,第二列的值
Cell c11 = rs.getCell(1, 1);
String strc11 = c11.getContents();
System.out.println("Cell(0, 0)" + " value : " + strc00 + "; type : " + c00.getType());
System.out.println("Cell(1, 0)" + " value : " + strc10 + "; type : " + c10.getType());
System.out.println("Cell(1, 1)" + " value : " + strc11 + "; type : " + c11.getType());
第四步:操作數據
如果僅僅是取得Cell的 值,我們可以方便地通過getContents()方法,它可以將任何類型的Cell值都作爲一個字符串返回。如果有需要知道Cell內容的確切類型,API也提供了一系列的方法:
String strc00 = null;
double strc10 = 0.00;
Date strc11 = null;
Cell c00 = rs.getCell(0, 0);
Cell c10 = rs.getCell(1, 0);
Cell c11 = rs.getCell(1, 1);
if(c00.getType() == CellType.LABEL)
{
LabelCell labelc00 = (LabelCell)c00;
strc00 = labelc00.getString();
}
if(c10.getType() == CellType.NUMBER)
{
NmberCell numc10 = (NumberCell)c10;
strc10 = numc10.getValue();
}
if(c11.getType() == CellType.DATE)
{
DateCell datec11 = (DateCell)c11;
strc11 = datec11.getDate();
}
System.out.println("Cell(0, 0)" + " value : " + strc00 + "; type : " + c00.getType());
System.out.println("Cell(1, 0)" + " value : " + strc10 + "; type : " + c10.getType());
System.out.println("Cell(1, 1)" + " value : " + strc11 + "; type : " + c11.getType());
在得到Cell對象後,通過 getType()方法可以獲得該單元格的類型,然後與API提供的基本類型相匹配,強制轉換成相應的類型,最後調用相應的取值方法getXXX(),就可以得到確定類型的值。
循環取出全部數據,並轉化爲相應格式:
int rows = sheet.getRows();
for (int i = 1; i < rows; i++) {
Cell cb1 = sheet.getCell(0, i);
Cell cb2 = sheet.getCell(1, i);
Cell num3 = sheet.getCell(2, i);
Cell num4 = sheet.getCell(3, i);
String user = "";
String rule = "";
int numNew = 0;
int numEdit = 0;
if (cb1.getType() == CellType.LABEL) {
LabelCell lc = (LabelCell) cb1;
user = lc.getString();
}
if (cb2.getType() == CellType.LABEL) {
LabelCell lc = (LabelCell) cb2;
rule = lc.getString();
}
if (num3.getType() ==CellType.NUMBER_FORMULA) {
NumberFormulaCell nc = (NumberFormulaCell)num3;
try {
numNew =Double.valueOf(nc.getFormula()).intValue();
} catch (FormulaException e) {
e.printStackTrace();
}
}
if (num4.getType() ==CellType.NUMBER_FORMULA) {
NumberFormulaCell nc =(NumberFormulaCell) num4;
try {
numEdit =Double.valueOf(nc.getFormula()).intValue();
} catch (FormulaException e) {
e.printStackTrace();
}
}
}
第五步:關閉對象,釋放內存。
完成對Excel電子表格數據的處理後,一定要使用close()方法來關閉先前創建的對象,以釋放讀取數據表的過程中所佔用的內存空間,在讀取大量數據時顯得尤爲重要。