`
renjie120
  • 浏览: 241086 次
  • 性别: Icon_minigender_1
  • 来自: 上海
博客专栏
D11bba82-ec4a-3d31-a3c0-c51130c62f1c
Java应用集锦
浏览量:23279
社区版块
存档分类
最新评论

java应用集锦8:使用poi进行excel操作,同时支持excel2003和2007

    博客分类:
  • java
阅读更多

下面的这段代码仅仅支持excel2003,要同时支持2007和2003的excel请看后半部分!!

经常使用exlce的操作,以前工作使用的jxl,现在工作又使用poi.还好以前总结过,轻松搞定.这里再次把代码贴出来,可以直接使用.

package poi;

import java.util.List;

/**
 * 测试poi的工具类. 
 * 
 */
public class PoiUtil {
	public static void main(String[] args) {
		String fileName = "D:\\工作\\开发需求\\测试2.xls";
                //参数1:文件名
                //参数2:要读取的多个sheet的index
                //参数3:是否在遇到第一个空行时停止往下读取
                List<String[][]> strs1 = ExcelReader.readAllExcel(fileName, new String[]{"3","1","2"}, true);
		for (int i = 0; i < strs1.size(); i++) {
			System.out.println(getString2Array(strs1.get(i)));
		} 
	}

	public static String getString2Array(String[][] str) {
		StringBuffer buf = new StringBuffer("[");
		//System.out.println("行数:" + str.length);
		//System.out.println("列数:" + str[0].length);
		int len1 = str.length;
		int len2 = str[0].length;
		for (int i = 0; i < len1; i++) {
			for (int j = 0; j < len2; j++) {
				buf.append(str[i][j]);
				if (j != len2 - 1) {
					buf.append(",");
				} else {
					buf.append("\n");
				}
			}
		}
		buf.append("]");
		return buf.toString();
	} 
}

 ExcelReader.java(仅对2003的excel操作有效!)

package poi;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFDateUtil;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;

public class ExcelReader {
    /**
     * 工作簿
     */
    private HSSFWorkbook workbook;

    /**
     * 日志记录
     */
    private Log log = LogFactory.getLog("logger");

    /**
     * excel表
     */
    private HSSFSheet sheet;

    /**
     * excel文件流
     */
    private FileInputStream fis;

    /**
     * 消息
     */
    private StringBuffer msg = null;


    public ExcelReader(File exl) throws IOException {
            fis = new FileInputStream(exl);
            workbook = new HSSFWorkbook(fis);
            msg = new StringBuffer();
    } 
    
	public ExcelReader(String exlFileName) throws IOException {
		File file = new File(exlFileName);
		fis = new FileInputStream(file);
		workbook = new HSSFWorkbook(fis);
		msg = new StringBuffer();
	}
    
	public ExcelReader(InputStream input) throws IOException {
		workbook = new HSSFWorkbook(input);
		msg = new StringBuffer();
	}

    public void destory() {
        try {
                msg = null;
                if(fis!=null)
                	fis.close();
        } catch (Exception ex) {
                log.error("ExcelReader-destory", ex);
                msg.append(ex.getMessage());
        }
    }

    public boolean setCurrentSheet(int num) {
            if (workbook != null && num < workbook.getNumberOfFonts()) {
                    try {
                            sheet = workbook.getSheetAt(num);
                            return true;
                    } catch (NullPointerException e) {
                            log.error("ExcelReader-setCurrentSheet", e);
                    }
            }        
            return false;
    }

    /**
     * 可以看到根据下面的五个参数确定了一个excel文件页面里面的一个矩形区域。
     * 
     * @param sheetNum
     *            文件的页面
     * @param firstRowNum
     *            第一行的行数
     * @param lastRowNum
     *            最后一行的行数
     * @param firstColIndex
     *            第一列的列数
     * @param lastColIndex
     *            最后一列的列数
     * @return
     */
    public String[][] getSheetAsTable(int sheetNum, int firstRowNum,
                    int lastRowNum, int firstColIndex, int lastColIndex) {
            String[][] cells = null;
            if (setCurrentSheet(sheetNum)) {
                    cells = new String[lastRowNum - firstRowNum + 1][lastColIndex
                                    - firstColIndex + 1];
                    int row = 0;
                    for (int c1 = firstRowNum; c1 <= lastRowNum; c1++) {
                            for (int c2 = firstColIndex; c2 <= lastColIndex; c2++) {
                                    try {
                                            cells[c1][c2] = getCellAsStringByIndex(c1, c2);
                                    } catch (Exception e) {
                                            log.error("ExcelReader-getSheetAsTable", e);
                                            cells[c1][c2] = "";
                                    }
                            }
                    }
            }
            return cells;
    }

    /**
     * 返回指定位置的单元格
     * 
     * @param rowId
     *            设置单元格的行
     * @param colId
     *            设置单元格的列
     * @return
     */
    public String getCellAsStringByIndex(int rowId, int colId) {
            String cellStr = "";
            if (sheet != null && rowId < sheet.getLastRowNum() + 1) {
                    try {
                            HSSFRow row = sheet.getRow(rowId);
                            if (row != null) {
                                    if (colId < row.getLastCellNum()) {
                                            HSSFCell cell = row.getCell((short) colId);
                                            if (cell != null) {
                                                    try {
                                                            switch (cell.getCellType()) {
                                                            case 0: {// CELL_TYPE_NUMERIC
                                                        if (HSSFDateUtil.isCellDateFormatted(cell)) { 
                					     Date d= HSSFDateUtil.getJavaDate(
                                   				  cell.getNumericCellValue());
                					  SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
                 				 		   cellStr = df.format(d);
            						}else{
                                                                    cellStr =Double.toString(cell.getNumericCellValue());   
             }
                                                                    break;
                                                            }
                                                            case 1: {// CELL_TYPE_STRING
                                                                    cellStr = cell.getStringCellValue();
                                                                    break;
                                                            }
                                                            case 2: {
                                                                    String formula = cell.getCellFormula();
                                                                    if (formula.indexOf("DATE(") >= 0) {
                                                                            Date d= HSSFDateUtil.getJavaDate(
                                                                                            cell.getNumericCellValue());
                                                                            SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
                                                                            cellStr = df.format(d);
                                                                    } else if (formula.indexOf("SUM(") >= 0) {
                                                                            cellStr = Double.toString(cell
                                                                                            .getNumericCellValue());
                                                                    }else if (formula.indexOf("SIN(") >= 0) {
                                                                            cellStr = Double.toString(cell
                                                                                            .getNumericCellValue());
                                                                    }
                                                                    else {
                                                                            cellStr = cell.getStringCellValue();
                                                                    }
                                                                    break;
                                                            }
                                                            case 4: {
                                                                    cellStr = Boolean.toString(cell.getBooleanCellValue());
                                                                    break;
                                                            }
                                                            default: {
                                                                    cellStr = new String("");
                                                            }
                                                                    if (cellStr == null) {
                                                                            cellStr = "";
                                                                    }
                                                            }
                                                    } catch (Exception e) {
                                                            log.error("ExcelReader-getCellAsStringByIndex", e);
                                                            cellStr = "";
                                                    }
                                            }
                                    }
                            }
                    } catch (Exception e) {
                            log.error("ExcelReader-getCellAsStringByIndex", e);
                            cellStr = "";
                    }
            }
            return cellStr;
    }
    
    /**
     * 返回指定sheet,指定行,指定列的单元格内容.
     * @param sheetNum
     * @param rowindex
     * @param colIndex
     * @return
     */
    public String getCellAsStringByIndex(int sheetNum,int rowindex,int colIndex)
    {
            if(setCurrentSheet(sheetNum)){
                    return getCellAsStringByIndex(rowindex,colIndex);
            }
            return "";
    }
    
    public void emptyCell(HSSFCell cell){
    	System.out.println(cell.getCellType());
    }
    
    public String getErrorMessage() {       
            return msg.toString();  
    }

    /**
     * 返回当前的页面
     * @return
     */
    public HSSFSheet getSheet() {
            return sheet;
    }

    /**
     * 返回当前的工作簿
     * @return
     */
    public HSSFWorkbook getWorkbook() {
            return workbook;
    }
    
    /**
	 * 
	 * @param fileName
	 *            要读取的excel文件名
	 * @param sheetNum
	 *            要读取的表单的数目
	 * @param row
	 *            要读取的单元格行数
	 * @param col
	 *            要读取的单元格列数
	 * @return
	 */
	public static String readExcel(String fileName, String sheetNum,
			String row, String col) { 
		try {
			ExcelReader excelRd = new ExcelReader(fileName);
			excelRd.setCurrentSheet(new Integer(sheetNum).intValue() - 1);
			return excelRd.getCellAsStringByIndex(
					(new Integer(row).intValue() - 1),
					(new Integer(col).intValue() - 1));
		} catch (Exception e) {
			e.printStackTrace();
			return "出现异常,可能是文件未找到!";
		}
	}

	/**
	 * 读取excel文件的一个表格里面的基本信息:最大行数
	 * 
	 * @param fileName
	 *            文件名
	 * @param sheetNum
	 *            要读取的表单的数目
	 * @return
	 */
	public static String readRowInfo(String fileName, String sheetNum) { 
		try {
			ExcelReader excelRd = new ExcelReader(fileName);
			excelRd.setCurrentSheet(new Integer(sheetNum).intValue() - 1);
			HSSFSheet sheet = excelRd.getSheet();
			int rowNum = sheet.getLastRowNum() + 1;
			return new Integer(rowNum).toString();
		} catch (Exception e) {
			e.printStackTrace();
			return "检查输入的文件是否存在或者页面不存在!";
		}
	}

	/**
	 * 读取excel中某一个表单的某一行的最大列数
	 * 
	 * @param fileName
	 *            文件名
	 * @param sheetNum
	 *            表单的数目
	 * @param row
	 *            行数
	 * @return
	 */
	public static String readColInfo(String fileName, String sheetNum,
			String row) { 
		ExcelReader excelRd;
		HSSFSheet sheet;
		HSSFRow rowNum;
		int colNum;
		try {
			excelRd = new ExcelReader(fileName);
			excelRd.setCurrentSheet(new Integer(sheetNum).intValue() - 1);
			sheet = excelRd.getSheet();
			rowNum = sheet.getRow(new Integer(row).intValue());
			colNum = rowNum.getLastCellNum();
			return new Integer(colNum).toString();
		} catch (Exception e) {
			e.printStackTrace();
			return "检查输入的文件是否存在!";
		}
	}

	/**
	 * 获取文件的工作簿的数目
	 * 
	 * @param fileName
	 *            文件名
	 * @return
	 */
	public static String readSheetInfo(String fileName) { 
		ExcelReader excelRd;
		HSSFWorkbook workbook;
		int sheetNum;
		try {
			excelRd = new ExcelReader(fileName);
			workbook = excelRd.getWorkbook();
			sheetNum = workbook.getNumberOfSheets();
			return new Integer(sheetNum).toString();
		} catch (Exception e) {
			e.printStackTrace();
			return "检查输入的文件是否存在!";
		}
	}

	/**
	 * 返回指定文件的页面的全部数据
	 * 
	 * @param fileName
	 *            文件名
	 * @param sheetNum
	 *            文件的表数
	 * @return String[][]
	 */
	public static String[][] readAllExcel(String fileName, String sheetNum) {
		return readAllExcel(fileName,sheetNum,false);
	}
	 
	public static List<String[][]> readAllExcel(String fileName, String[] sheetNums) {
		return readAllExcel(fileName,sheetNums,false);
	}
	
	public static List<String[][]> readAllExcel(String fileName, String[] sheetNums,
			boolean returnMeetFirstNullRow) {  
		ExcelReader excelRd;
		try {
			excelRd = new ExcelReader(fileName);
			return readAllExcel(excelRd, sheetNums, returnMeetFirstNullRow);
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}

	/**
	 * 返回指定文件的页面的全部数据,并指定是否在第一行遇到null的时候自动停止继续往下搜索 
	 * @param fileName
	 * @param sheetNum
	 * @param returnFirstNullRow
	 *            遇到第一个空行自动停止继续往下搜索.
	 * @return
	 */
	public static String[][] readAllExcel(String fileName, String sheetNum,
			boolean returnMeetFirstNullRow) {  
		ExcelReader excelRd;
		try {
			excelRd = new ExcelReader(fileName);
			return readAllExcel(excelRd, sheetNum, returnMeetFirstNullRow);
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
	
	/**
	 * 返回指定文件的页面的全部数据
	 * @param input
	 * @param sheetNum
	 * @return
	 */
	public static String[][] readAllExcel(InputStream input, String sheetNum) {
		return readAllExcel(input,sheetNum,false);
	}
	
	/**
	 * 私有方法:返回指定excel中的指定sheet内容.
	 * @param excelRd
	 * @param sheetNum
	 * @param returnMeetFirstNullRow
	 * @return
	 */
	private static String[][] readAllExcel(ExcelReader excelRd, String sheetNum,
			boolean returnMeetFirstNullRow) {  
		String[] sheetNums = new String[]{sheetNum};
		List<String[][]> ans  = readAllExcel(excelRd,sheetNums,returnMeetFirstNullRow);
		return ans.get(0);
	}
	
	/**
	 * 一次返回多个sheet的内容.
	 * @param excelRd
	 * @param sheetNums
	 * @param returnMeetFirstNullRow
	 * @return
	 */
	private static List<String[][]> readAllExcel(ExcelReader excelRd, String[] sheetNums,
			boolean returnMeetFirstNullRow) {  
		//如果遇到第一个空行自动返回,调用下面的方法.
		List<String[][]> ans = new ArrayList<String[][]>(); 
		int sheetId;
		HSSFSheet sheet;
		int maxRowNum;
		HSSFRow row;
		int maxColNum;
		String[][] result;
		try { 
			for(int i=sheetNums.length-1;i>=0;i--){
				result = new String[][] {{}};
				sheetId = new Integer(sheetNums[i]).intValue() - 1;
				excelRd.setCurrentSheet(sheetId);
				sheet = excelRd.getSheet();
				//如果设置了要遇到第一个空行就自动返回,就计算maxRowNum!
				if (returnMeetFirstNullRow) {
					maxRowNum = 0;
					row = null;
					// 得到从0行开始的第一个非空行数.
					for (;; maxRowNum++) {
						row = sheet.getRow(maxRowNum);
						if (row == null) {
							if (maxRowNum != 0)
								maxRowNum--; 
							break;
						}
					}
				}
				//否则直接取全部的excel的行数
				else {
					maxRowNum = sheet.getLastRowNum();  
				} 
				if(maxRowNum!=0){
					// 得到第一行的列数.
					row = sheet.getRow(0);
					maxColNum = row.getLastCellNum();
					result = excelRd.getSheetAsTable(sheetId, 0, maxRowNum, 0,
							maxColNum - 1);
				}
				ans.add(result); 
			}
			excelRd.destory();
			return ans;
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}

	/**
	 * 返回指定文件的页面的全部数据
	 * @param input
	 * @param sheetNum
	 * @param returnMeetFirstNullRow 是否遇到空行直接返回
	 * @return
	 */
	public static String[][] readAllExcel(InputStream input, String sheetNum,
			boolean returnMeetFirstNullRow) {
		ExcelReader excelRd;
		try {
			excelRd = new ExcelReader(input);
			return readAllExcel(excelRd, sheetNum, returnMeetFirstNullRow);
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
	
	/**
	 * 返回全部的指定sheet中的exlce内容.	 
	 * @param input
	 * @param sheetNums
	 * @return
	 */
	public static List<String[][]> readAllExcel(InputStream input, String[] sheetNums) {
		return readAllExcel(input,sheetNums,false);
	}
	
	/**
	 * 返回全部的指定sheet中的exlce内容.
	 * @param input
	 * @param sheetNums sheet字符串数组
	 * @param returnMeetFirstNullRow
	 * @return
	 */
	public static List<String[][]> readAllExcel(InputStream input,
			String[] sheetNums, boolean returnMeetFirstNullRow) {
		ExcelReader excelRd;
		try {
			excelRd = new ExcelReader(input);
			return readAllExcel(excelRd, sheetNums, returnMeetFirstNullRow);
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
}

 ExcelWrite.java

package poi;

import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.List;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFCellStyle;
import org.apache.poi.hssf.usermodel.HSSFFont;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.hssf.util.HSSFColor;

public class ExcelWrite {
    /**
     * 工作簿
     */
    private HSSFWorkbook workbook;

    /**
     * 日志记录
     */
    private Log log = LogFactory.getLog("logger");

    /**
     * excel表
     */
    private HSSFSheet sheet;

    /**
     * excel文件流
     */
    private FileInputStream fis;

    /**
     * 消息
     */
    private StringBuffer msg = null;

    public ExcelWrite()
    {
            
    }
    /**
     * 从一个javaBean的集合向目标文件写数据
     * @param data 一个装有javaBean的list
     * @param targetFile 目标excel文件
     */
    public String setSheetFromTable(List data, String targetFile) {
            try {
                    HSSFWorkbook targetWorkbook = new HSSFWorkbook();
                    FileOutputStream fout = new FileOutputStream(targetFile,true);
                    setDataToWorksheet(data, targetWorkbook, 0);
                    targetWorkbook.write(fout);
                    fout.flush();
                    fout.close();
                    return "ok";
            } catch (Exception e) {
                    log.error("出现异常", e);
                    return "";
            }
    }

    /**
     * 将list的数据放进excel的一个工作表中去。
     * @param data 数据的来源list,是一个来自于数据库的装满了javabean的list
     * @param workbook 目的excel工作簿
     * @param sheetNum 目的excel工作簿的表格要填写数据的页码
     */
    public void setDataToWorksheet(List data,HSSFWorkbook workbook,int sheetNum)
    {
            HSSFRow title = null;
            HSSFSheet sheet = null;
            try{
            if(data.size()<1)
            {
                    return ;
            }
            sheet = workbook.createSheet();
            
            //下面设置cell的文字格式
            font = workbook.createFont();
            font.setFontName(HSSFFont.FONT_ARIAL);
            font.setUnderline((byte)1);
            font.setColor(HSSFColor.BLUE.index);
            //下面设置标题行的样式                
            titleStyle = workbook.createCellStyle();
            titleStyle.setBorderBottom((short)1);
            titleStyle.setBorderLeft((short)1);
            titleStyle.setBorderRight((short)1);
            titleStyle.setBorderTop((short)1);
            titleStyle.setFont(font);
            titleStyle.setFillForegroundColor(HSSFColor.LIGHT_CORNFLOWER_BLUE.index);
            titleStyle.setFillPattern(HSSFCellStyle.SOLID_FOREGROUND);
            
            //取list中的第一个数据,进行属性名称的读取,准备放到excel表格中的第一行
            Object aData = data.get(0);
            PropertyDescriptor[] props = Introspector.getBeanInfo(
                            aData.getClass(), Object.class).getPropertyDescriptors();
            //在表格的第一行建立一个数据行,用来放置这些属性的名称
            title = sheet.createRow(0);
            //设置行高.注意值设置的很大。。
            title.setHeight((short)500);
            for(short temp = 0; temp < props.length; temp++)
            {        HSSFCell cell = title.createCell(temp);
                    cell.setCellStyle(titleStyle);
                    cell.setCellType(HSSFCell.CELL_TYPE_STRING);
                    cell.setCellValue(props[temp].getShortDescription());
                    //设置各个列的宽度
                    sheet.setColumnWidth((short)temp, (short)5000);
            }
            for(int temp = 0;temp<data.size();temp++)
            {
                    //实际的数据是开始从第二行开始进行传递的
                    HSSFRow row = sheet.createRow(temp+1);
                    //取出javabean对象里面的各个属性的值
                    Object obj = data.get(temp);
                    String values[] = getPropertyOfBean(obj);
                    for(short cellNum=0;cellNum<values.length;cellNum++){
                            HSSFCell cell = row.createCell(cellNum);
                            cell.setCellType(HSSFCell.CELL_TYPE_STRING);
                            cell.setCellValue(values[cellNum]);
                    }
            }
            }catch(Exception e)
            {
                    log.error("出现bug",e);
            }
    }

    HSSFFont font = null;
    HSSFCellStyle titleStyle = null;
    /**
     * 设置excel表格的样式
     *
     */
    private void setStyle()
    {
            
    }
    
    /**
     * 根据一个javabean对象,返回这个对象的属性值集合,使用到反射机制。
     * @return
     */
    private String[] getPropertyOfBean(Object obj) {
            String[] result = null;
            try {
                    PropertyDescriptor[] props = Introspector.getBeanInfo(
                                    obj.getClass(), Object.class).getPropertyDescriptors();
                    result = new String[props.length];
                    for (int temp = 0; temp < props.length; temp++) {
                            try {
                                    result[temp] = props[temp].getReadMethod().invoke(obj)
                                                    .toString();
                            } catch (Exception e) {
                                    log.error("出现异常", e);
                                    return null;
                            }
                    }
            } catch (Exception e1) {
                    log.error("出现异常", e1);
                    return null;
            }
            return result;
    }
}

 

 

同时支持excel2003和excel2007的java类:

对excel2003和excel2007使用的java类是不一样的,对于2003使用的java类前缀一般为HSSF,例如HSSFWorkbook,HSSFCell等,而对于2007前缀是XSSF,例如XSSFWorkbook类等.

下面的这个类创建一个workBook对象,并根据excel文件自动进行判断需要实例化2003的excel解析类还是2007的excel的解析类:

 

 
package poi;

import java.io.IOException;
import java.io.InputStream;
import java.io.PushbackInputStream;

import org.apache.poi.POIXMLDocument;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
import org.apache.poi.openxml4j.opc.OPCPackage;
import org.apache.poi.poifs.filesystem.POIFSFileSystem;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;

public class WorkbookFactory {
	/**
	* Creates an HSSFWorkbook from the given POIFSFileSystem
	*/
	public static Workbook create(POIFSFileSystem fs) throws IOException {
	   return new HSSFWorkbook(fs);
	}
	/**
	* Creates an XSSFWorkbook from the given OOXML Package
	*/
	public static Workbook create(OPCPackage pkg) throws IOException {
	   return new XSSFWorkbook(pkg);
	} 
	
	public static Workbook create(InputStream inp) throws IOException, InvalidFormatException {
	    if(! inp.markSupported()) {
	    inp = new PushbackInputStream(inp, 8);
	   }
	  
	   if(POIFSFileSystem.hasPOIFSHeader(inp)) {
	    return new HSSFWorkbook(inp);
	   }
	   if(POIXMLDocument.hasOOXMLHeader(inp)) {
	    return new XSSFWorkbook(OPCPackage.open(inp));
	   }
	   throw new IllegalArgumentException("Your InputStream was neither an OLE2 stream, nor an OOXML stream");
	}
	}
	 

 

 

然后在调用生成workbook对象的时候,一律使用超类型即可,不要再使用HSSFWorkBook或者XSSFWorkBook的类型就行了!

例如下面的例子:ExcelReader.java,将其中的所有的构造函数中修改使用上面的create方法创建返回workBook对象即可!并且去掉所有前缀含有HSSF的类名即可!!

 

 

public ExcelReader(File exl) throws IOException {
            fis = new FileInputStream(exl);
            workbook = WorkbookFactory .create(fis);
            msg = new StringBuffer();
    } 
    
	public ExcelReader(String exlFileName) throws IOException {
		File file = new File(exlFileName);
		fis = new FileInputStream(file);
		workbook = WorkbookFactory .create(fis);
		msg = new StringBuffer();
	}
    
	public ExcelReader(InputStream input) throws IOException {
		workbook =WorkbookFactory .create(fis);
		msg = new StringBuffer();
	}
 

 

在实践中,遇到了这样的一个异常:

java.lang.NegativeArraySizeException
        at org.apache.poi.hssf.record.SSTDeserializer.readStringRemainder(SSTDeserializer.java:335)
        at org.apache.poi.hssf.record.SSTDeserializer.processContinueRecord(SSTDeserializer.java:320)
        at org.apache.poi.hssf.record.SSTRecord.processContinueRecord(SSTRecord.java:539)
        at org.apache.poi.hssf.record.RecordFactory.createRecords(RecordFactory.java:216)
        at org.apache.poi.hssf.usermodel.HSSFWorkbook.<init>(HSSFWorkbook.java:181)
        at org.apache.poi.hssf.usermodel.HSSFWorkbook.<init>(HSSFWorkbook.java:228)
        at org.apache.poi.hssf.usermodel.HSSFWorkbook.<init>(HSSFWorkbook.java:209)
 找了很久,结果换成使用最新的poi包,以及修改使用上面的create()创建新的workBook对象就可以了(后来不改这个类也可以,仅仅换包就行了),有点奇怪,不知道怎么引起的...

分享到:
评论
2 楼 hngmduyi 2013-05-18  
1 楼 米奇风 2011-10-21  
作者代码风格很规范,这篇文章很好,消除大多数人在开发过程中的疑惑,一针见血

相关推荐

    C#语言 EXCEL和SQL数据交互集锦

    3. 使用NPOI库:NPOI是针对.NET平台的Java POI库的移植,支持.xls和.xlsx格式,适用于大量数据处理。 二、SQL数据的访问 1. ADO.NET:微软提供的标准数据库访问技术,包括SqlConnection、SqlCommand、...

    URL汇集、jsp、java学习中各种问题集锦

    根据提供的标题、描述以及部分链接内容,我们可以整理出与JSP、Java开发相关的若干...以上内容基于给定的信息进行了整理和扩展,旨在提供关于JSP和Java开发过程中的实用技巧和解决方案。希望这些信息能对读者有所帮助。

    2023年通信施工应急预案.doc

    2023年通信施工应急预案.doc

    234751_43.m3u8

    234751_43.m3u8

    ARM单片机与FPGA协同驱动的先进运动控制卡方案:实时控制功能与应用拓展一体化解决方案

    内容概要:本文详细介绍了基于ARM单片机(LM3S6911)和FPGA(ALTERA EP1C3)的运动控制卡设计方案。ARM负责复杂的插补算法和以太网通信,FPGA则承担实时脉冲生成和IO扩展任务。文中展示了具体的硬件连接、通信协议以及关键代码片段,如以太网通信的LWIP协议栈实现、FPGA的Verilog代码用于脉冲生成和IO消抖处理。此外,还讨论了硬件设计细节,如PCB布局和电源管理,强调了系统的实时性和抗干扰能力。 适合人群:对嵌入式系统和运动控制系统感兴趣的工程师和技术爱好者,尤其是有一定ARM和FPGA开发经验的人群。 使用场景及目标:适用于工业自动化领域的运动控制应用,旨在提高运动控制的精度和实时性,减少机械抖动,增强系统的可靠性和灵活性。 其他说明:本文不仅提供了详细的硬件和软件设计思路,还附带了原理图、PCB图及源代码,方便读者进行进一步的研究和开发。

    Paddle Serving-部署一个自己的OCR识别服务器

    Paddle Serving-部署一个自己的OCR识别服务器

    COMSOL空气流注放电模型——集成针-针电极及Helmholtz光电离过程等离子体模拟新范式

    内容概要:本文详细介绍了如何利用COMSOL的等离子体模块构建针-针电极间的空气流注放电模型。主要内容涵盖了几何结构的定义、物理场配置(如电子、正负离子的载流子选择)、化学反应的设定(包括21组带电粒子反应)以及Helmholtz光电离过程的具体实现方法。文中还提供了多个代码片段用于解释各个步骤的操作方式,并强调了求解器配置和边界条件处理的关键点。此外,作者分享了一些实用的小技巧,如初始步长设置、网格细化等,以确保模型能够稳定收敛并得到合理的仿真结果。 适合人群:从事等离子体物理研究的专业人士,特别是那些对高压放电现象感兴趣的科研工作者和技术人员。 使用场景及目标:适用于希望深入了解和模拟针-针电极间空气流注放电行为的研究项目。通过该模型可以更好地理解电场分布、粒子密度变化等微观物理过程,从而为实际工程应用提供理论支持。 阅读建议:由于涉及较多的技术细节和数学公式,建议读者具备一定的电磁学、流体力学基础知识,并且最好有一定的COMSOL软件使用经验。同时,在实践中可以根据自己的研究方向调整模型参数进行探索。

    西门子S7-200 PLC与触摸屏实现的恒压供水系统全套图纸程序:一拖二与一拖三配置的实践应用

    内容概要:本文详细介绍了基于西门子S7-200 PLC的恒压供水系统的设计与实现。系统采用一拖二或一拖三模式,确保供水系统的可靠性。核心组件包括PLC、富士PID控制模块和ABB变频器,通过精确的压力控制和快速响应,实现了稳定的恒压供水。文中提供了详细的PLC程序示例,涵盖水泵启停控制、PID算法调用以及变频器频率调节等功能。此外,还展示了触摸屏界面设计,用于实时监控和操作。硬件配置方面,强调了柜体制作图纸的重要性和规范性,确保电气接线正确无误。调试过程中,作者分享了许多实用技巧,如PID参数整定、变频器设置和故障切换逻辑等。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是对PLC编程和恒压供水系统感兴趣的读者。 使用场景及目标:适用于工业现场的恒压供水系统设计与实施,旨在提高供水系统的稳定性和可靠性,减少因设备故障导致的生产中断。 其他说明:本文不仅提供了完整的硬件配置和软件编程指导,还分享了丰富的实战经验和调试技巧,帮助读者更好地理解和应用相关技术。

    基于模型预测算法的三电平整流器输入不平衡控制MATLAB仿真模型研究

    内容概要:本文详细介绍了基于MATLAB的三电平整流器输入不平衡控制的仿真模型。该模型采用模型预测控制(MPC)和正负序分离技术,通过Clarke变换和Park变换将三相电压转换到αβ坐标系,并进一步分离出正序和负序分量。随后,通过模型预测控制计算网侧参考电流,确保系统在网侧电压不平衡情况下仍能稳定运行。文中还讨论了仿真过程中的一些关键技术细节,如代价函数设计、LCL滤波器的谐振频率设置以及动态调整预测步长的方法。最终,仿真结果显示该方法在电网电压不平衡时表现出色,直流侧电压纹波控制在2%以内,电流波形质量良好。 适合人群:从事电力电子、电力系统控制领域的研究人员和技术人员,特别是对三电平整流器及其控制策略感兴趣的读者。 使用场景及目标:适用于研究和开发三电平整流器在非理想电网条件下的控制策略,旨在提高系统的稳定性和效率,减少谐波和直流侧电压波动。 其他说明:文中提供了详细的MATLAB代码片段和调试技巧,有助于读者理解和实现该仿真模型。此外,作者还分享了一些实际调试中的经验和注意事项,如坐标变换的时序对齐问题和代价函数权重的选择。

    Asp.Net CRM系统:高效管理客户关系,实现商业价值的最佳选择和必备工具

    内容概要:本文详细介绍了Asp.Net CRM客户关系管理系统的功能和技术实现。该系统不仅涵盖了客户信息管理、日程安排等功能,还展示了如何通过C#代码实现这些功能的具体细节。此外,文章强调了系统的二次开发能力和扩展性,如通过创建新的数据库表和编写相应代码来满足特定行业需求。系统采用三层架构,将客户生命周期、销售流程、团队协同等功能模块有机结合,形成一个完整的业务协同平台。文中还探讨了销售流程引擎、数据关联、实时消息推送等方面的技术实现,突出了系统的灵活性和高效性。 适合人群:对CRM系统开发感兴趣的软件工程师、企业IT管理人员、有一定编程基础的研发人员。 使用场景及目标:适用于希望提升客户管理水平、优化销售流程、增强内部协作的企业。通过实施该系统,企业可以更好地管理客户资源,提高工作效率,降低成本,最终提升销售业绩。 其他说明:文章提供了丰富的代码示例,帮助读者深入理解系统的工作原理和技术实现。同时,强调了系统在实际应用中的灵活性和扩展性,使其能够适应不同企业的具体需求。

    PowerShell7.5.1

    PowerShell7.5.1

    FX5U以太网通讯超清

    内容概要:本文详细介绍了三菱FX5U PLC的以太网通讯配置和编程技巧。首先讲解了基本的硬件配置和参数设置,如IP地址、子网掩码、端口号等。接着展示了如何使用ST语言和Python进行TCP客户端编程,包括创建Socket、连接服务器、发送和接收数据的具体步骤。文中还提到了一些常见的陷阱和解决方案,如隐藏寄存器的设置、十六进制地址转换、连接超时等问题。此外,提供了高级玩法,如批量读写寄存器、使用MC协议读取数据以及通过Python脚本测试通讯稳定性。最后强调了调试工具Wireshark的使用和注意事项。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是对三菱PLC以太网通讯感兴趣的读者。 使用场景及目标:帮助读者掌握三菱FX5U PLC以太网通讯的配置和编程方法,提高设备联机和数据采集效率,解决实际项目中可能遇到的各种问题。 其他说明:文章不仅提供了详细的代码示例,还分享了许多实用的经验和技巧,有助于读者快速上手并避免常见错误。

    PMSM永磁同步电机双环矢量控制理论深度解析与仿真实践指南

    内容概要:本文详细介绍了永磁同步电机(PMSM)的双环矢量控制理论及其在MATLAB/Simulink中的具体实现方法。首先解释了电流环和速度环的工作原理,特别是电流环中的Clarke变换和Park变换的具体实现。文中提供了详细的代码示例,如Clarke变换、Park变换以及电流环PI控制器的实现。同时,针对常见的调试问题给出了具体的解决方案,如速度环PI参数的整定、SVPWM模块的配置等。此外,还强调了仿真过程中需要注意的关键点,如PWM载波频率、死区时间补偿、速度观测器的选择等。 适合人群:从事电机控制研究和技术开发的专业人士,尤其是有一定MATLAB/Simulink基础的研发人员。 使用场景及目标:适用于希望深入了解PMSM双环矢量控制原理并掌握其仿真的技术人员。目标是在MATLAB/Simulink环境中搭建稳定的PMSM双环控制系统,确保系统在不同负载条件下的稳定性和鲁棒性。 其他说明:文章不仅提供了理论分析,还包括了大量的实践经验分享,帮助读者避免常见错误,提高仿真效率。

    FX3U PLC控制器资料:尺寸、主控芯片、电源及功能详解

    内容概要:本文深入介绍了FX3U PLC控制器的硬件架构和嵌入式软件设计。硬件方面,详细描述了控制器的尺寸、主控芯片STM32F103VCT6及其外围电路设计,如电源管理、继电器输出、光耦隔离、模拟量处理和通讯接口(CAN总线、RS485)。软件部分则展示了关键的GPIO配置、ADC校准、中断处理、状态指示灯控制以及多种通讯协议的实现方法。文中还特别强调了抗干扰设计和工业应用场景中的优化措施。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是对PLC控制器硬件设计和嵌入式编程感兴趣的读者。 使用场景及目标:适用于希望深入了解PLC控制器内部工作原理的技术人员,帮助他们掌握如何设计和优化工业控制系统,特别是在小型产线控制器或智能仓储系统的开发中。 其他说明:文章不仅提供了详细的硬件和软件设计方案,还分享了许多实践经验,如光耦隔离的应用、抗干扰措施、以及针对特定工业环境的优化技巧。

    2025年软件实施工程师笔试面试题及答案.docx

    2025年软件实施工程师笔试面试题及答案.docx

    CAD技术在水利水电工程中的应用.docx

    CAD技术在水利水电工程中的应用.docx

    可编程设置±10V、±5V、±3V高精度正负电压输出模块特点:小体积、宽电压输入范围、自动保存电压值、板载LDO稳定及低成本方案

    内容概要:本文介绍了一种低成本、高精度的模拟量正负电压输出模块方案,适用于工业控制和实验室设备。该模块支持±10V、±5V、±3V可编程设置,具备断电参数保存、自定义校准等功能。硬件设计包括电源模块、DAC芯片、EEPROM和控制芯片,确保输出电压稳定性和高精度。软件部分实现了电压设置、参数保存和自定义校准功能。材料成本控制在15元左右,性价比极高。 适合人群:电子工程师、硬件开发者、工业控制系统设计师以及对高精度电压输出有需求的研究人员。 使用场景及目标:① 工业控制系统的精密电压调节;② 实验室设备的高精度电压输出;③ 自动化项目的批量部署。 其他说明:该方案提供了完整的硬件设计和软件实现,包括详细的代码示例和原理图,方便用户进行二次开发和定制。

    淘客云全开源淘客系统:源码开放、支持定制开发,一站式搭建部署与首年免费更新支持

    内容概要:本文深入剖析了一个全开源淘客系统的实现细节,涵盖了多个关键技术点。首先介绍了用Go语言重构的淘宝联盟API网关,展示了商品搜索接口的核心代码,强调了参数签名、HTTP请求处理以及数据转换的重要性。接着讨论了数据库设计,特别是佣金结算表的结构,突出了JSON类型的使用和乐观锁机制的应用。前端方面,展示了基于Vue.js的佣金日历组件,体现了数据驱动UI和事件冒泡处理的设计理念。部署环节则分享了一些实用技巧,如手动更新淘宝API的SSL证书链。此外,还探讨了PHP实现的订单同步服务、SDK封装、热更新通道等特性,以及系统的技术售后支持措施。最后,文章总结了该系统的优点,如清晰的MVC分层结构、灵活的分佣规则设计、完善的部署方案和技术售后支持。 适合人群:对淘客系统开发感兴趣的开发者,尤其是希望深入了解系统架构设计、前后端开发、数据库设计和部署优化的技术人员。 使用场景及目标:适用于想要搭建或改进淘客系统的团队和个人开发者。目标是帮助读者理解淘客系统的各个组成部分及其工作原理,从而能够快速上手并进行二次开发。 其他说明:文中提供了大量实际代码片段和配置示例,便于读者理解和实践。同时,作者还分享了许多实用的经验和技巧,有助于提高开发效率和系统稳定性。

    LabVIEW 2018版:新手必备的动态启动界面制作教程,涉及二维图片写入、控件移动与动态调用知识点

    内容概要:本文详细介绍了如何使用 LabVIEW 2018 版制作动态启动界面,涵盖三大核心技术:图片加载、控件移动和动态调用。首先,通过创建透明图像和使用相对路径确保跨平台兼容性,实现了高质量的图片展示。其次,利用 While 循环和属性节点精确控制控件的移动轨迹,如 Logo 飞入、文字滑入等效果。最后,采用 VI 动态调用技术,使主程序在启动界面前台动画播放完毕后无缝加载,提升了用户体验。此外,文中还提供了许多优化建议,如双缓冲模式、渐隐效果、状态机架构等,帮助开发者打造更加专业的启动界面。 适合人群:具有一定 LabVIEW 基础的开发者,尤其是希望提升用户界面美观度和交互体验的技术人员。 使用场景及目标:适用于需要开发带有动态启动界面的 LabVIEW 应用程序,旨在提高软件的专业性和视觉吸引力,同时确保启动过程的流畅性和稳定性。 其他说明:文中提供的代码片段和技巧可以直接应用于实际项目中,建议初学者逐步尝试各个功能模块,熟悉后再进行综合应用。

    STM32驱动的信号发生器:基于Matlab数据生成与DMA传输的波形输出系统,支持串口通信与TFTLCD显示,一键切换波形,附程序源码与硬件原理图

    内容概要:本文详细介绍了如何利用STM32单片机和Matlab协作构建一个高效的信号发生器。首先,通过Matlab生成不同类型的波形数据并将其转化为适用于DAC的12位格式,然后将这些数据存储为C语言数组以便嵌入STM32程序中。接着,在STM32端配置DAC通道并通过DMA进行高速数据传输,确保波形能够稳定地输出。此外,文中还探讨了串口通信协议的设计用于远程控制波形参数,如频率、幅度等;实现了按键消抖功能以实现波形类型的手动切换;并且优化了LCD显示屏的内容刷新机制。最终测试表明该系统可以在1Hz-50kHz范围内提供高质量的波形输出,总谐波失真控制在1%以内。 适合人群:具有一定嵌入式开发经验的技术人员,尤其是对STM32、Matlab有一定了解的人群。 使用场景及目标:适用于需要定制化波形输出的应用场合,如实验教学、产品研发测试等。主要目的是为了帮助开发者更好地理解和掌握STM32与外部设备交互的方式,特别是DAC与DMA配合使用的技巧。 其他说明:随附有完整的项目资料包,包括Matlab脚本、Keil工程项目文件以及硬件原理图等,方便读者实际操作验证。

Global site tag (gtag.js) - Google Analytics