POI Excel 合并数据相同的行

简介: import java.io.Serializable;/** * POI Excel报表导出,列合并实体 * * @author WQ * */public class PoiModel implements Serializable{ /** * ...
import java.io.Serializable;

/**
 * POI Excel报表导出,列合并实体<br>
 * 
 * @author WQ
 *
 */
public class PoiModel implements Serializable{
    /**
     * 
     */
    private static final long serialVersionUID = 1L;

    private String content;  
      
    private String oldContent;  
  
    private int rowIndex;  
  
    private int cellIndex;

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }

    public String getOldContent() {
        return oldContent;
    }

    public void setOldContent(String oldContent) {
        this.oldContent = oldContent;
    }

    public int getRowIndex() {
        return rowIndex;
    }

    public void setRowIndex(int rowIndex) {
        this.rowIndex = rowIndex;
    }

    public int getCellIndex() {
        return cellIndex;
    }

    public void setCellIndex(int cellIndex) {
        this.cellIndex = cellIndex;
    }

    public PoiModel() {
    }

    public PoiModel(String content, String oldContent, int rowIndex,
            int cellIndex) {
        this.content = content;
        this.oldContent = oldContent;
        this.rowIndex = rowIndex;
        this.cellIndex = cellIndex;
    }

    @Override
    public String toString() {
        return "PoiModel [content=" + content + ", oldContent=" + oldContent
                + ", rowIndex=" + rowIndex + ", cellIndex=" + cellIndex + "]";
    }  
  
}

 

 

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import org.apache.poi.ss.usermodel.Cell;
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.ss.util.CellRangeAddress;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;

public class Demo {

    /** 
     * @param title 标题集合 tilte的长度应该与list中的model的属性个数一致 
     * @param maps 内容集合 
     * @param mergeIndex 合并单元格的列 
     */  
    public static String createExcel(String[] title, Map<String, List<Map<String, String>>> maps, int[] mergeIndex){  
        if (title.length==0){  
            return null;  
        }  
        /*初始化excel模板*/  
        Workbook workbook = new XSSFWorkbook();  
        Sheet sheet = null;  
        int n = 0;  
        for(Map.Entry<String, List<Map<String, String>>> entry : maps.entrySet()){  
            try {  
                sheet = workbook.createSheet();  
                workbook.setSheetName(n, entry.getKey());  
                workbook.setSelectedTab(0);  
            }catch (Exception e){  
                e.printStackTrace();  
            }  
            /*初始化head,填值标题行(第一行)*/  
            Row row0 = sheet.createRow(0);  
            for(int i = 0; i<title.length; i++){  
                /*创建单元格,指定类型*/  
                Cell cell_1 = row0.createCell(i, Cell.CELL_TYPE_STRING);  
                cell_1.setCellValue(title[i]);  
            }  
            /*得到当前sheet下的数据集合*/  
            List<Map<String, String>> list = entry.getValue();  
            /*遍历该数据集合*/  
            List<PoiModel> poiModels = new ArrayList();  
            if(null!=workbook){  
                Iterator iterator = list.iterator();  
                int index = 1;
                while (iterator.hasNext()){  
                    Row row = sheet.createRow(index);  
                    /*取得当前这行的map,该map中以key,value的形式存着这一行值*/  
                    Map<String, String> map = (Map<String, String>)iterator.next();  
                    /*循环列数,给当前行塞值*/  
                    for(int i = 0; i<title.length; i++){  
                        String old = "";  
                        /*old存的是上一行统一位置的单元的值,第一行是最上一行了,所以从第二行开始记*/  
                        if(index > 1){  
                            old = poiModels.get(i)==null?"":poiModels.get(i).getContent();  
                        }  
                        /*循环需要合并的列*/  
                        for(int j = 0; j < mergeIndex.length; j++){  
                            if(index == 1){  
                                /*记录第一行的开始行和开始列*/  
                                PoiModel poiModel = new PoiModel();  
                                poiModel.setOldContent(map.get(title[i]));  
                                poiModel.setContent(map.get(title[i]));  
                                poiModel.setRowIndex(1);  
                                poiModel.setCellIndex(i);  
                                poiModels.add(poiModel);  
                                break;  
                            }else if(i > 0 && mergeIndex[j] == i){/*这边i>0也是因为第一列已经是最前一列了,只能从第二列开始*/  
                                /*当前同一列的内容与上一行同一列不同时,把那以上的合并, 或者在当前元素一样的情况下,前一列的元素并不一样,这种情况也合并*/  
                                /*如果不需要考虑当前行与上一行内容相同,但是它们的前一列内容不一样则不合并的情况,把下面条件中
                  ||poiModels.get(i).getContent().equals(map.get(title[i])) && !poiModels.get(i - 1).getOldContent().equals(map.get(title[i-1]))去掉就行
*/ if(!poiModels.get(i).getContent().equals(map.get(title[i])) || poiModels.get(i).getContent().equals(map.get(title[i]))
                    && !poiModels.get(i - 1).getOldContent().equals(map.get(title[i-1]))){ /*当前行的当前列与上一行的当前列的内容不一致时,则把当前行以上的合并*/ CellRangeAddress cra=new CellRangeAddress(poiModels.get(i).getRowIndex(), index - 1, poiModels.get(i).getCellIndex(), poiModels.get(i).getCellIndex()); //在sheet里增加合并单元格 sheet.addMergedRegion(cra); /*重新记录该列的内容为当前内容,行标记改为当前行标记,列标记则为当前列*/ poiModels.get(i).setContent(map.get(title[i])); poiModels.get(i).setRowIndex(index); poiModels.get(i).setCellIndex(i); } } /*处理第一列的情况*/ if(mergeIndex[j] == i && i == 0 && !poiModels.get(i).getContent().equals(map.get(title[i]))){ /*当前行的当前列与上一行的当前列的内容不一致时,则把当前行以上的合并*/ CellRangeAddress cra=new CellRangeAddress(poiModels.get(i).getRowIndex(), index - 1, poiModels.get(i).getCellIndex(), poiModels.get(i).getCellIndex()); //在sheet里增加合并单元格 sheet.addMergedRegion(cra); /*重新记录该列的内容为当前内容,行标记改为当前行标记*/ poiModels.get(i).setContent(map.get(title[i])); poiModels.get(i).setRowIndex(index); poiModels.get(i).setCellIndex(i); } /*最后一行没有后续的行与之比较,所有当到最后一行时则直接合并对应列的相同内容*/ if(mergeIndex[j] == i && index == list.size()){ CellRangeAddress cra=new CellRangeAddress(poiModels.get(i).getRowIndex(), index, poiModels.get(i).getCellIndex(), poiModels.get(i).getCellIndex()); //在sheet里增加合并单元格 sheet.addMergedRegion(cra); } } Cell cell = row.createCell(i, Cell.CELL_TYPE_STRING); cell.setCellValue(map.get(title[i])); /*在每一个单元格处理完成后,把这个单元格内容设置为old内容*/ poiModels.get(i).setOldContent(old); } index++; } } n++; } /*生成临时文件*/ FileOutputStream out = null; String localPath = null; File tempFile = null; String fileName = String.valueOf(new Date().getTime()/1000); try { tempFile = File.createTempFile(fileName, ".xlsx"); localPath = tempFile.getAbsolutePath(); out = new FileOutputStream(localPath); workbook.write(out); }catch (IOException e){ e.printStackTrace(); }finally { try { out.flush(); out.close(); }catch (IOException e){ e.printStackTrace(); } } return localPath; } public static void main(String[] args) throws IOException{ /*此处标题的数组则对应excel的标题*/ String[] title = {"id","标题","描述","负责人","开始时间"}; List<Map<String, String>> list = new ArrayList(); /*这边是制造一些数据,注意每个list中map的key要和标题数组中的元素一致*/ for(int i = 0; i<10; i++){ HashMap<String, String> map = new HashMap(); if(i > 5){ if(i<7){ map.put("id","333"); map.put("标题","mmmm"); }else { map.put("id","333"); map.put("标题","aaaaa"); } }else if (i >3){ map.put("id","222"); map.put("标题","哈哈哈哈"); }else if (i>1 && i<3){ map.put("id","222"); map.put("标题","hhhhhhhh"); }else { map.put("id","222"); map.put("标题","bbbb"); } map.put("描述","sssssss"); map.put("负责人","vvvvv"); map.put("开始时间","2017-02-27 11:20:26"); list.add(map); } Map<String, List<Map<String, String>>> map = new HashMap(); map.put("测试合并数据", list); System.out.println(createExcel(title, map, new int[]{0,1,2})); } }

 

思路:

后来就通过建一个单元格对象,属性有单元格内容,与单元格同一列的上一行内容,起始行,起始列,就这四个属性

 

生成的文件效果(两种情况的):

 

 
 
相关文章
|
21天前
|
数据采集 数据可视化 数据挖掘
利用Python自动化处理Excel数据:从基础到进阶####
本文旨在为读者提供一个全面的指南,通过Python编程语言实现Excel数据的自动化处理。无论你是初学者还是有经验的开发者,本文都将帮助你掌握Pandas和openpyxl这两个强大的库,从而提升数据处理的效率和准确性。我们将从环境设置开始,逐步深入到数据读取、清洗、分析和可视化等各个环节,最终实现一个实际的自动化项目案例。 ####
|
2月前
|
数据采集 存储 JavaScript
自动化数据处理:使用Selenium与Excel打造的数据爬取管道
本文介绍了一种使用Selenium和Excel结合代理IP技术从WIPO品牌数据库(branddb.wipo.int)自动化爬取专利信息的方法。通过Selenium模拟用户操作,处理JavaScript动态加载页面,利用代理IP避免IP封禁,确保数据爬取稳定性和隐私性。爬取的数据将存储在Excel中,便于后续分析。此外,文章还详细介绍了Selenium的基本设置、代理IP配置及使用技巧,并探讨了未来可能采用的更多防反爬策略,以提升爬虫效率和稳定性。
153 4
|
4月前
|
关系型数据库 MySQL Shell
不通过navicat工具怎么把查询数据导出到excel表中
不通过navicat工具怎么把查询数据导出到excel表中
52 0
|
2月前
|
数据处理 Python
Python实用记录(十):获取excel数据并通过列表的形式保存为txt文档、xlsx文档、csv文档
这篇文章介绍了如何使用Python读取Excel文件中的数据,处理后将其保存为txt、xlsx和csv格式的文件。
98 3
Python实用记录(十):获取excel数据并通过列表的形式保存为txt文档、xlsx文档、csv文档
|
2月前
|
easyexcel Java UED
SpringBoot中大量数据导出方案:使用EasyExcel并行导出多个excel文件并压缩zip后下载
在SpringBoot环境中,为了优化大量数据的Excel导出体验,可采用异步方式处理。具体做法是将数据拆分后利用`CompletableFuture`与`ThreadPoolTaskExecutor`并行导出,并使用EasyExcel生成多个Excel文件,最终将其压缩成ZIP文件供下载。此方案提升了导出效率,改善了用户体验。代码示例展示了如何实现这一过程,包括多线程处理、模板导出及资源清理等关键步骤。
|
2月前
|
前端开发 JavaScript Java
导出excel的两个方式:前端vue+XLSX 导出excel,vue+后端POI 导出excel,并进行分析、比较
这篇文章介绍了使用前端Vue框架结合XLSX库和后端结合Apache POI库导出Excel文件的两种方法,并对比分析了它们的优缺点。
799 0
|
2月前
|
Java Apache
Apache POI java对excel表格进行操作(读、写) 有代码!!!
文章提供了使用Apache POI库在Java中创建和读取Excel文件的详细代码示例,包括写入数据到Excel和从Excel读取数据的方法。
59 0
|
3月前
|
数据采集 存储 数据挖掘
使用Python读取Excel数据
本文介绍了如何使用Python的`pandas`库读取和操作Excel文件。首先,需要安装`pandas`和`openpyxl`库。接着,通过`read_excel`函数读取Excel数据,并展示了读取特定工作表、查看数据以及计算平均值等操作。此外,还介绍了选择特定列、筛选数据和数据清洗等常用操作。`pandas`是一个强大且易用的工具,适用于日常数据处理工作。
|
4月前
|
SQL JSON 关系型数据库
n种方式教你用python读写excel等数据文件
n种方式教你用python读写excel等数据文件
|
4月前
|
存储 Java Apache