Java使用POI解析后缀为.xls和.xlsx的Excel的完美处理

在我们做web开发中,经常性的于Excel表格进行打交道,但其中的坑只有我们自己去做了才知道,坑是有多大。当Excel表格中使用了公式计算数据的时候,我们用POI中的CELL_TYPE_FORMULA这个类型只能拿到类似于这样的公式:SUM(P4-Q4-R4-S4),那怎么获取他的值呢,接下来,请看以下demo


import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;

import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellValue;
import org.apache.poi.ss.usermodel.DateUtil;
import org.apache.poi.ss.usermodel.FormulaEvaluator;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;

/**
 * 
 * @author arryluo
 * 
 */
public class Texst {
	private static final String EXCEL_XLS = "xls";
	private static final String EXCEL_XLSX = "xlsx";
	private static XSSFWorkbook xssfWorkbook = null;
	private static HSSFWorkbook workbook = null;

	public static void main(String[] args) {
		// 判断后缀是xls,还是xlsx
		String path = "d:/user.xls";
		File file = new File(path);
		if (checkFile(file)) {
			// 进行后缀为xlsx的Excel操作
			xlsx(file);
		} else {
			// 进行xls的操作
			xls(file);
		}
	}

	// 检查该文件是否是xls,还是xlsx类型
	private static boolean checkFile(File file) {
		boolean flg = false;
		if (file.exists()) {

			String name = file.getName();
			// String lastindex = name.substring(name.lastIndexOf(".") + 1);
			if (name.endsWith(EXCEL_XLSX))
				flg = true;
			else
				flg = false;
		} else {
			System.out.println("文件不存在");
		}
		return flg;
	}

	// 如果是xlsx,做什么操作
	private static void xlsx(File file) {

		try {
			xssfWorkbook = new XSSFWorkbook(new FileInputStream(file));
			show(xssfWorkbook);
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

	}

	//他们两者都有相同的父类,而且都是父类中的方法操作,那我干脆点直接将他们用父类的形式抽取出来,这样解决了代码的冗余问题
	private static void show(Workbook workbook) {
		// 循环工作表Sheet
		for (int numSheet = 0; numSheet < workbook.getNumberOfSheets(); numSheet++) {
			Sheet xssfSheet = workbook.getSheetAt(numSheet);
			// 这种主要是判断是否含空以免包空指针
			if (xssfSheet == null) {
				continue;
			}

			// 循环行Row
			// 从第三行读取(这个根据自己的实际需求而定,因为我是在第三行之后才是我需要的数据)
			for (int rowNum = 3; rowNum <= xssfSheet.getLastRowNum(); rowNum++) {
				Row xssfRow = xssfSheet.getRow(rowNum);
				if (xssfRow == null) {
					continue;
				}
				// 循环列Cell
				System.out.println("多少列" + xssfRow.getLastCellNum());
				for (int cellNum = 0; cellNum <= xssfRow.getLastCellNum(); cellNum++) {
					Cell xssfCell = xssfRow.getCell(cellNum);
					if (xssfCell == null) {
						continue;
					}
					System.out.println(getValue(xssfCell, workbook));
				}
				System.out.print("******");
			}
		}
	}

	// 如果是就做什么操作
	private static void xls(File file) {
		try {
			InputStream inputStream = new FileInputStream(file);
			workbook = new HSSFWorkbook(inputStream);
			show(workbook);
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

	@SuppressWarnings("static-access")
	// 将他们两者的父类抽取出来,这样减少代码量,同时便于管理
	private static String getValue(Cell xssfCell, Workbook workbook) {
		String value = "";
		switch (xssfCell.getCellType()) {
		case Cell.CELL_TYPE_STRING:
			value = String.valueOf(xssfCell.getRichStringCellValue()
					.getString());
			System.out.print("|");
			break;
		case Cell.CELL_TYPE_NUMERIC:
			if (DateUtil.isCellDateFormatted(xssfCell)) {
				value = String.valueOf(String.valueOf(xssfCell
						.getDateCellValue()));
			} else {
				value = String.valueOf(xssfCell.getNumericCellValue());
			}
			System.out.print("|");
			break;
		case Cell.CELL_TYPE_BOOLEAN:
			value = String.valueOf(xssfCell.getBooleanCellValue());
			System.out.print("|");
			break;
		// 公式,
		case Cell.CELL_TYPE_FORMULA:
			// 获取Excel中用公式获取到的值,//=SUM(P4-Q4-R4-S4)Excel用这种类似的公式计算出来的值用POI无法获取,要想获取的话,就必须一下操作
			FormulaEvaluator evaluator = workbook.getCreationHelper()
					.createFormulaEvaluator();
			evaluator.evaluateFormulaCell(xssfCell);
			CellValue cellValue = evaluator.evaluate(xssfCell);
			value = String.valueOf(cellValue.getNumberValue());
			break;
		case Cell.CELL_TYPE_ERROR:
			value = String.valueOf(xssfCell.getErrorCellValue());
			break;
		default:
		}
		return value;
	}
}

架包链接:

链接:http://pan.baidu.com/s/1i4DGVuX 密码:6xrw

  • 4
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值