POI最新版本 4.1.2 操作 Excel

本文章仅用于个人学习之用,不做他用,侵权请告知删除。

原文章链接:https://www.jianshu.com/p/45dc02506b3e
文章作者:DeadBoty

1. POI 简介

POI(Poor Obfuscation Implementation),直译为“可怜的模糊实现”,利用POI接口可以通过 Java 操作 Microsoft office 套件工具的读写功能。POI支持office的所有版本。

POI 的 Maven 依赖

<!-- 操作以 .xls 为后缀的 Excel -->
<dependency>
    <groupId>org.apache.poi</groupId>
    <artifactId>poi</artifactId>
    <version>4.1.2</version>
</dependency>
<!-- 操作以 .xlsx 为后缀的 Excel -->
<dependency>
    <groupId>org.apache.poi</groupId>
    <artifactId>poi-ooxml</artifactId>
    <version>4.1.2</version>
</dependency>

2. POI 的快速应用

(1)操作后缀为 .xlsExcel 文件

在 POI 中的几个主要对象:

此方法适用于以 .xls 结尾的 Excel 2003 之前的版本

HSSFWorkbookExcel 工作簿workbook
HSSFSheetExcel 工作表 sheet
HSSFRowExcel 行
HSSFCellExcel 单元格
使用 POI 创建 Excel

使用 POI 创建 Excel

/**
  * 创建 Excel 测试
  * @throws IOException 文件操作异常
  */
@Test
public void createTest() throws IOException {
    // 新建工作薄
    HSSFWorkbook workbook = new HSSFWorkbook();
    // 新建工作表
    HSSFSheet sheet = workbook.createSheet();
    // 创建行,行号为 0
    HSSFRow row = sheet.createRow(0);
    // 创建单元格,表示为第 3 个单元格
    HSSFCell cell = row.createCell(2);
    // 输入值
    cell.setCellValue("Hello World!");
    // 将 Excel 进行文件导出
    // 获取文件路径和文件
    Path path = Paths.get(".").resolve("hello.xls");
    // 文件已存在则删除
    if (Files.exists(path)){
        Files.delete(path);
    }
    // 创建文件
    File file = Files.createFile(path).toFile();
    // 写入数据
    workbook.write(file);
}

运行结果:

img

使用 POI 向 Excel 中追加

使用 POI 向 Excel 中追加

/**
  * 向已存在的 Excel 中追加内容
  * @throws IOException
  */
@Test
public void appendTest() throws IOException {
    // 获取文件输入流
    String path = "./hello.xls";
    File file = new File(path);
    int rowIndex = 4;
    HSSFWorkbook workbook = null;
    HSSFSheet sheet = null;
    if(file.exists()){
        FileInputStream fis = new FileInputStream(file);
        // 新建工作簿
        workbook = new HSSFWorkbook(fis);
        // 新建工作表
        sheet = workbook.getSheet("sheet0");
    } else{
        // 新建工作簿
        workbook = new HSSFWorkbook();
        // 新建工作表
        sheet = workbook.createSheet();
    }
    // 获取行号
    HSSFRow row = sheet.getRow(rowIndex);
    // 判断是否存在,不存在则创建
    if (row == null){
        row = sheet.createRow(rowIndex);
    }
    // 创建单元格,表示为第 3 个单元格
    HSSFCell cell = row.createCell(5);
    // 输入值
    cell.setCellValue("追加3!");
    // 将 Excel 进行文件导出
    // 写入数据
    workbook.write(file);
}

运行结果:

img

使用 POI 读取 Excel

使用 POI 读取 Excel

/**
  * 读取 Excel 测试
  * @throws IOException 文件操作异常
  */
@Test
public void readTest() throws IOException {
    // 读取 Excel 文件
    // 获取文件路径和文件
    FileInputStream fis = new FileInputStream("./hello.xls");
    // 将输入流转换为工作簿对象
    HSSFWorkbook workbook = new HSSFWorkbook(fis);
    // 获取第一个工作表
    HSSFSheet sheet = workbook.getSheet("sheet0");
    // 使用索引获取工作表
    // HSSFSheet sheet = workbook.getSheetAt(0);
    // 获取指定行
    HSSFRow row = sheet.getRow(0);
    // 获取指定列
    HSSFCell cell = row.getCell(2);
    // 打印
    System.out.println(cell.getStringCellValue());
}

运行结果:

img

(2)操作后缀为 .xlsxExcel 文件

在 POI 中的几个主要对象:

此方法适用于以 .xlsx 结尾的 Excel 2007 之后的版本

XSSFWorkbookExcel 工作簿workbook
XSSFSheetExcel 工作表 sheet
XSSFRowExcel 行
XSSFCellExcel 单元格
使用 POI 创建 Excel 表

使用 POI 创建 Excel 表

/**
  * 创建以 .xlsx 为后缀的 Excel 测试
  * @throws IOException 文件操作异常
  */
@Test
public void createXlsxTest() throws IOException {
    // 新建工作簿
    XSSFWorkbook workbook = new XSSFWorkbook();
    // 新建工作表
    XSSFSheet sheet = workbook.createSheet();
    // 创建行,行号为 0
    XSSFRow row = sheet.createRow(0);
    // 创建单元格,表示为第 3 个单元格
    XSSFCell cell = row.createCell(2);
    // 输入值
    cell.setCellValue("Hello World!");
    // 将 Excel 进行文件导出
    // 获取文件
    File file = new File("./hello.xlsx");
    // 如果文件存在则删除
    if (file.exists()){
        file.delete();
    }
    // 创建输出流
    FileOutputStream fos = new FileOutputStream(file);
    // 写入
    workbook.write(fos);
    fos.close();
}

运行结果:

img

使用 POI 追加 Excel 文件

使用 POI 追加 Excel 文件

/**
  * 向已存在以 .xlsx 为后缀的 Excel 中追加内容
  * @throws IOException
  */
@Test
public void appendXlsxTest() throws IOException {
    // 获取文件输入流
    String path = "./hello.xlsx";
    File file = new File(path);
    int rowIndex = 4;
    XSSFWorkbook workbook = null;
    XSSFSheet sheet = null;
    if(file.exists()){
        FileInputStream fis = new FileInputStream(file);
        // 新建工作簿
        workbook = new XSSFWorkbook(fis);
        // 新建工作表
        sheet = workbook.getSheet("sheet0");
    } else{
        // 新建工作簿
        workbook = new XSSFWorkbook();
        // 新建工作表
        sheet = workbook.createSheet();
    }
    // 创建行,行号为 0
    XSSFRow row = sheet.getRow(rowIndex);
    // 判断是否存在,不存在则创建
    if (row == null){
        row = sheet.createRow(rowIndex);
    }
    // 创建单元格,表示为第 3 个单元格
    XSSFCell cell = row.createCell(5);
    // 输入值
    cell.setCellValue("追加内容1");
    // 将 Excel 进行文件导出
    FileOutputStream fos = new FileOutputStream(file);
    // 写入数据
    workbook.write(fos);
}

运行结果:

img

使用 POI 读取 Excel 文件

使用 POI 读取 Excel 文件

/**
  * 读取以 .xlsx 为后缀的 Excel 测试
  * @throws IOException 文件操作异常
  */
@Test
public void readXlsxTest() throws IOException {
    // 读取 Excel 文件
    // 获取文件路径和文件
    FileInputStream fis = new FileInputStream("./hello.xlsx");
    // 将输入流转换为工作簿对象
    XSSFWorkbook workbook = new XSSFWorkbook(fis);
    // 获取第一个工作表
    XSSFSheet sheet = workbook.getSheet("sheet0");
    // 使用索引获取工作表
    // XSSFSheet sheet = workbook.getSheetAt(0);
    // 获取指定行
    XSSFRow row = sheet.getRow(0);
    // 获取指定列
    XSSFCell cell = row.getCell(2);
    // 打印
    System.out.println(cell.getStringCellValue());
}

运行结果:

img

(3)根据后缀名读取 Excel 文件

通过正则表达式读取

通过正则表达式读取

/**
  * 根据后缀名读取 Excel 文件
  */
@Test
public void readAllExcel() throws IOException {
    // 两个不同后缀名文件的路径
    String path1 = "./hello.xlsx";
    String path2 = "./hello.xls";
    readExcel(path1);
    readExcel(path2);
}

/**
  * 根据文件路径读取 Excel 文件
  * @param path 文件路径
  * @throws IOException 文件操作异常
  */
public void readExcel(String path) throws IOException {
    // 获取后缀
    String suffix = path.substring(path.lastIndexOf("."));
    // 判断后缀为 xls 还是 xlsx
    if (path.matches("^.+\\.(?i)((xls)|(xlsx))$")){
        // 创建输入流
        FileInputStream fis = new FileInputStream(path);
        // 判断是否为以 .xls 结尾的 Excel 文件
        boolean isXlsExcel = path.matches("^.+\\.(?i)(xls)$");
        // 判断后缀生成 工作簿
        Workbook workbook = isXlsExcel ? new HSSFWorkbook(fis) : new XSSFWorkbook(fis);
        // 获取工作表
        Sheet sheet = workbook.getSheet("sheet0");
        // 获取指定行
        Row row = sheet.getRow(0);
        // 获取指定列
        Cell cell = row.getCell(2);
        // 输出值
        System.out.println(suffix + " :\t" + cell.getStringCellValue());

    }
}

运行结果:

img

通过 WorkbookFactory 读取

通过 WorkbookFactory 读取

/**
 * 通用读取的方法
 */
@Test
public void generalRead() throws IOException {
    String path1 = "./hello.xls";
    String path2 = "./hello.xlsx";
    generalRead(path1);
    generalRead(path2);
}

public void generalRead(String path) throws IOException {
    File file = new File(path);
    Workbook workbook = WorkbookFactory.create(file);
    Sheet sheet = workbook.getSheet("sheet0");
    String value = sheet.getRow(0).getCell(2).getStringCellValue();
    String suffix = path.substring(path.lastIndexOf(".")+1);
    System.out.println(suffix + " :\t" + value);
}

运行结果:

img

3. POI 的高级操作

(1)合并单元格

/**
  * 合并单元格
  * CellRangeAddress(firstRow,lastRow,firstCol,lastCol)
  * 四个参数分别为 起使行号,终止行号,起使列,终止列
  *
  * @throws IOException 文件操作异常
  */
@Test
public void mergeCells() throws IOException {
    // 新建工作簿
    HSSFWorkbook workbook = new HSSFWorkbook();
    // 新建工作表
    HSSFSheet sheet1 = workbook.createSheet("sheet1");
    // 设置要合并的单元格范围,合并第二行的第二列到第五列为一个单元格
    CellRangeAddress rangeAddress = new CellRangeAddress(1, 1, 1, 4);
    sheet1.addMergedRegion(rangeAddress);
    // 设置合并后的单元格内容
    HSSFRow row = sheet1.createRow(1);
    HSSFCell cell = row.createCell(1);
    cell.setCellValue("我是合并后的单元格");
    FileOutputStream outputStream = new FileOutputStream("./工作簿.xls");
    workbook.write(outputStream);
    outputStream.close();
}

运行结果:

img

**注:**上图中合并单元格后,单元格的名称是第一个单元格;即上面中合并了第二行的第二列到第五列,合并后的单元格叫 B2,而其它被合并的单元格已经无效了,不能对无效单元格设置值。如果进行了设置将不显示。

(2)设置单元格样式

/**
  * 设置单元格样式
  * @throws IOException 文件操作异常
  */
@Test
public void cellStyle() throws IOException {
    // 新建工作簿
    HSSFWorkbook workbook = new HSSFWorkbook();
    // 新建工作表
    HSSFSheet sheet = workbook.createSheet("工作表1");
    // 创建行
    HSSFRow row = sheet.createRow(3);
    // 创建单元格
    HSSFCell cell = row.createCell(3);
    cell.setCellValue("单元格内容");
    // 设置sheet的第四列的宽度, 列宽单位是字符的1 / 256.
    sheet.setColumnWidth(3, 20 * 256);
    // 创建单元格样式
    HSSFCellStyle style = workbook.createCellStyle();
    // 设置单元格内容水平居中
    style.setAlignment(HorizontalAlignment.CENTER);
    // 设置单元格内容垂直居中
    style.setVerticalAlignment(VerticalAlignment.CENTER);
    // 创建字体
    HSSFFont font = workbook.createFont();
    font.setFontName(HSSFFont.FONT_ARIAL);//字体为Arial
    font.setColor(HSSFFont.COLOR_RED);//字体颜色为红色
    font.setBold(true);//设置为粗体
    font.setFontHeightInPoints((short) 16);//设置字体大小
    style.setFont(font);
    // 设置填充模式,模式为全部前景色
    style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
    // 设置前景色为绿色
    style.setFillForegroundColor(HSSFColorPredefined.GREEN.getColor().getIndex());
    // 设置背景色,如果填充模式为其它填充模式,这个前景和背景色将相互交映显示
    style.setFillBackgroundColor(HSSFColorPredefined.RED.getColor().getIndex());
    // 为特定单元格设置样式
    cell.setCellStyle(style);
    FileOutputStream fos = new FileOutputStream("./工.xls");
    workbook.write(fos);
    fos.close();
}

运行结果:

img

使用 POI 导入 Excel 文件到 MySQL 中

使用 POI 导入 Excel 文件到 MySQL 中

// 使用 Spring Test 测试,数据库操作可更改为普通的 JDBC 操作
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:applicationContext.xml")
public class TestPoi {
    @Autowired
    private DataSource dataSource;

    @Test
    public void test() throws IOException, SQLException {
        // Excel 路径
        String path = "./xxx.xlsx";
        File file = new File(path);
        // 获取数据库连接
        Connection conn = dataSource.getConnection();
        // 插入的 SQL 语句
        String sql = "insert into sheet1 values(null,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)";
        PreparedStatement ps = conn.prepareStatement(sql);
        // 获取工作簿
        Workbook workbook = WorkbookFactory.create(file);
        // 获取当前表
        Sheet sheet = workbook.getSheetAt(0);
        // 获取总行数
        int totalRows = sheet.getPhysicalNumberOfRows();
        // 获取表头的总列数
        int totalCols = sheet.getRow(0).getPhysicalNumberOfCells();
        // 遍历行
        for (int i = 1; i < totalRows; i++) {
            // 遍历列
            for (int j = 0; j < totalCols; j++) {
                // 获取 i 行 j 列
                Cell cell = sheet.getRow(i).getCell(j);
                // 判断该列是否为 null
                if (cell == null || cell.getCellType() == CellType.BLANK){
                    ps.setString(j+1,null);
                    continue;
                }
                // 由于测试的 Excel 此处的最后一位为日期类型,这里使用日期输出
                if (j==totalCols-1){
                    cell.setCellType(CellType.NUMERIC);
                    System.out.println(new Date(cell.getDateCellValue().getTime()));
                    ps.setDate(j+1,new Date(cell.getDateCellValue().getTime()));
                }else {
                    // 判断是否是字符类型
                    if(cell.getCellType() == CellType.STRING){
                        ps.setString(j+1,cell.getStringCellValue());
                    }else{
                        // 单元格为数值类型
                        cell.setCellType(CellType.NUMERIC);
                        ps.setInt(j+1, (int) cell.getNumericCellValue());
                    }
                }
            }
            // 执行 SQL
            ps.executeUpdate();
        }
        ps.close();
        conn.close();
    }
}
java_POI教程.pdf javaPOI操作Excel文件.doc POI_API帮助文档.chm poi-bin-3.9-20121203.tar.gz poi帮助.docx POI中文帮助文档.pdf poi中文教程.doc第一章 POI简介 实际的开发中,表现层的解决方案虽然有多样,但是IE浏览器已成为最多人使用的浏览器,因为大家都用Windows。在企业办公系统中,常常有客户这样子要求:你要把我们的报表直接用Excel打开(电信系统、银行系统)。或者是:我们已经习惯用Excel打印。 workbook(HSSFWorkbook),一个workbook可以有多个sheet(HSSFSheet)组成,一个sheet是由多个row(HSSFRow)组成,一个row是由多个cell(HSSFCell)组成。 POI可以到www.apache.org下载到。实际运行时,需要有poi包就可以了。HSSF提供给用户使用的对象在rg.apache.poi.hssf.usermodel包中,主要部分包括Excel对象,样式和格式,还有辅助操作。有以下几种对象: HSSFWorkbook excel的文档对象 HSSFSheet excel的表单 HSSFRow excel的行 HSSFCell excel的格子单元 HSSFFont excel字体 HSSFDataFormat 日期格式 在poi1.7中才有以下2项: HSSFHeader sheet头 HSSFFooter sheet尾(只有打印的时候才能看到效果) 和这个样式 HSSFCellStyle cell样式 辅助操作包括 HSSFDateUtil 日期 HSSFPrintSetup 打印 HSSFErrorConstants 错误信息表 以下可能需要使用到如下的类 import org.apache.poi.hssf.usermodel.HSSFCell; import org.apache.poi.hssf.usermodel.HSSFCellStyle; import org.apache.poi.hssf.usermodel.HSSFDataFormat; 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; 先看poi的examples包中提供的最简单的例子,建立一个空xls文件。 import java.io.FileOutputStream; import java.io.IOException; import org.apache.poi.hssf.usermodel.HSSFWorkbook; public class ExcelSample1 { public static void main(String[] args) throws IOException { //创建一个excel文件 HSSFWorkbook wb= new HSSFWorkbook(); FileOutputStream fileOut= new FileOutputStream("c:\\workbook.xls"); // FileOutputStream fileOut= new FileOutputStream("c:/workbook.xls"); wb.write(fileOut); fileOut.close(); } } 通过这个例子,我们在c盘下建立的是一个空白的xls文件(不是空文件)。在此基础上,我们可以进一步看其它的例子。 import org.apache.poi.hssf.usermodel.*; import java.io.FileOutputStream; import java.io.IOException; public class CreateCells { public static void main(String[] args) throws IOException { HSSFWorkbook wb = new HSSFWorkbook(); //建立新HSSFWorkbook对象 HSSFSheet sheet = wb.createSheet("new sheet"); //建立新的sheet对象 HSSFRow row = sheet.createRow((short)0); //在sheet里创建一行,参数为行号(第一行,此处可想象成数组) HSSFCell cell = row.createCell((short)0); //在row里建立新cell(单元格),参数为列号(第一列) cell.setCellvalue(1); //设置cell的整数类型的值 row.createCell((short)1).setCellvalue(1.2); //设置cell浮点类型的值 row.createCell((short)2).setCellvalue("test"); //设置cell字符类型的值 row.createCell((short)3).setCellvalue(true); //设置cell布尔类型的值 HSSFCellStyle cellStyle = wb.createCellStyle(); //建立新的cell样式 cellStyle.setDataFormat(HSSFDataFormat. getBuiltinFormat("m/d/yy h:mm")); //设置cell样式为定制的日期格式 HSSFCell dCell =row.createCell((short)4); dCell.setCellvalue(new Date()); //设置cell为日期类型的值 dCell.setCellStyle(cellStyle); //设置该cell日期的显示格式 HSSFCell csCell =row.createCell((short)5); csCell.setEncoding(HSSFCell.ENCODING_UTF_16); //设置cell编码解决中文高位字节截断 csCell.setCellvalue("中文测试_Chinese Words Test"); //设置中西文结合字符串 row.createCell((short)6).setCellType(HSSFCell.CELL_TYPE_ERROR); //建立错误cell FileOutputStream fileOut = new FileOutputStream("workbook.xls"); wb.write(fileOut); fileOut.close(); } } 通过这个例子,我们可以清楚的看到xls文件从大到小包括了HSSFWorkbook HSSFSheet HSSFRow HSSFCell这样几个对象。我们可以在cell中设置各种类型的值。 尤其要注意的是如果你想正确的显示非欧美的字符时,尤其象中日韩这样的语言,必须设置编码为16位的即是HSSFCell.ENCODING_UTF_16,才能保证字符的高8位不被截断而引起编码失真形成乱码。 其他测试可以通过参考examples包中的测试例子掌握poi的详细用法,包括字体的设置,cell大小和低纹的设置等。需要注意的是POI是一个仍然在完善中的公开代码的项目,所以有些功能正在不断的扩充。 感觉上面的操作比较的繁琐,然后就自己写了一个方法。这个方法不需要事先创建row和cell,直接进行cteateCell就可以了,在程序中会自动进行判断,如果不存在的话会创建。 private static void cteateCell(HSSFWorkbook wb,HSSFRow row,short col,short align,String val){ HSSFCell cell = row.createCell(col); cell.setEncoding(HSSFCell.ENCODING_UTF_16); cell.setCellValue(val); HSSFCellStyle cellstyle = wb.createCellStyle(); cellstyle.setAlignment(align); cell.setCellStyle(cellstyle); } 对里面的几个参数的说明: short col 应该是你的cell单元格的位置也就是列号; short align 应该是你的对齐方式; String val 应该是你单元格里面要添加的值; 具体的调用如下: HSSFRow row = sheet.createRow((short)1); cteateCell(wb,row,(short)0,HSSFCellStyle.ALIGN_CENTER_SELECTION,"SampleID"); 在上边的例子里我们看到了要设置一个单元格里面信息的格式(例如,要将信息居中)设置的操作如下: HSSFCellStyle cellstyle = wb.createCellStyle(); cellstyle.setAlignment(HSSFCellStyle.ALIGN_CENTER_SELECTION); cell.setCellStyle(cellstyle); 还有我们我们经常会用到的合并单元格,在这里我们也有这样的操作,代码如下: sheet.addMergedRegion(new Region(1,(short)1,2,(short)4)); 这里面我们还要介绍一个经常会遇到的问题,就是怎么来冻结一个窗口。poi也为我们集成了这样的事情了。代码如下: sheet.createFreezePane(1,2);  在这里我们需要注意的是 一、 该方法是在一个具体的sheet里面来进行操作。 二、 方法createFreezepane;有2个参数。前一个参数代表列;后一个参数代表行。 上边的代码对应的excel文件如下: 我么在画面上看到了明显的两条黑线,这就是冻结的窗口。 然后我们来看一个完整的STRUTS的小例子,在这个例子里面我们要做的事情是要模拟移动公司的网上营业厅里面的一个功能,我们要把一个客户当月的通话记录和各种信息查询出来,并且生成一张excel报表。首先,我们来看一下网上效果的截图。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值