Java詳解如何將excel數(shù)據(jù)轉為樹形
前言
今天收到一個導入的任務,要求將excel數(shù)據(jù)保存到數(shù)據(jù)庫中,不同于普通的導入,這個導入的數(shù)據(jù)是一個樹形結構,如下圖:

通過觀察數(shù)據(jù)中的層級列我們發(fā)現(xiàn)表格數(shù)據(jù)由2棵樹組成,分別是第3,4,5,6,7,8,9,10,11和12,13,14,15,16,17,18,它們由0作樹的根節(jié)點,1為0的子節(jié)點,2為相鄰1的子節(jié)點,由此得出第一顆樹的結構為:

拆分原始數(shù)據(jù)
1.創(chuàng)建實體類
創(chuàng)建vo接收解析數(shù)據(jù),在這里,我們只關心層級屬性
@Excel(name = "層級")
private String hierarchy;
@Excel(name = "物料編碼")
private String materialCode;
@Excel(name = "物料名稱")
private String materialName;
@Excel(name = "基礎數(shù)量")
private BigDecimal materialNum;
@Excel(name = "使用數(shù)量")
private BigDecimal useAmount;
@Excel(name = "BOM版本")
private String version;
@Excel(name = "默認BOM")
private String isDefaults;2.處理數(shù)據(jù)
將數(shù)據(jù)源拆分為若干棵樹的數(shù)據(jù)集
代碼如下(示例):
/**
* 將集合對象按指定元素分割存儲
*
* @param materialVos 原始集合
* @param s 分割元素(這里是當集合對象層級為0時則分割,也就是樹的根節(jié)點為0)
* @return 每棵樹的結果集
*/
private List<List<MatMaterialBomImportVo>> subsection(List<MatMaterialBomImportVo> materialVos, String s) {
List<List<MatMaterialBomImportVo>> segmentedData = new ArrayList<>();
if (materialVos != null) {
//獲取指定元素的數(shù)量,判斷出最終將拆分為多少段
List<MatMaterialBomImportVo> collect = materialVos.stream().filter(bom -> s.equals(bom.getHierarchy())).collect(Collectors.toList());
int count = 0;
for (int i = 0; i < collect.size(); i++) {
List<MatMaterialBomImportVo> bomImportVo = new ArrayList<>();
boolean num = false;
//遍歷數(shù)據(jù)源
for (; count < materialVos.size(); count++) {
//第一個必然為樹的根節(jié)點,直接獲取并跳過
if (count == 0) {
bomImportVo.add(materialVos.get(count));
continue;
}
//當數(shù)據(jù)源第n個等于根節(jié)點并且已經(jīng)成功添加過數(shù)據(jù)時判斷為一段數(shù)據(jù)的結束,跳出循環(huán),
if (s.equals(materialVos.get(count).getHierarchy()) && num) {
break;
}
bomImportVo.add(materialVos.get(count));
num = true;
}
segmentedData.add(bomImportVo);
}
}
return segmentedData;
}手動設置每棵樹每個節(jié)點的id以及父id
代碼如下(示例):
for (List<MatMaterialBomImportVo> segmentedDatum : subsection(materialVos, "0")) {
//設置id以及父id
int i = 0;
for (MatMaterialBomImportVo vo : segmentedDatum) {
BeanTrim.beanAttributeValueTrim(vo);
vo.setPrimaryKey(i);
getParentId(vo, segmentedDatum);
i++;
}
}
/**
* 設置父id
*
* @param vo
* @param segmentedDatum
*/
private void getParentId(MatMaterialBomImportVo vo, List<MatMaterialBomImportVo> segmentedDatum) {
for (int j = vo.getPrimaryKey(); j >= 0; j--) {
if (Integer.parseInt(segmentedDatum.get(j).getHierarchy()) == Integer.parseInt(vo.getHierarchy()) - 1) {
vo.setForeignKey(segmentedDatum.get(j).getPrimaryKey());
break;
}
if (j == 0) {
vo.setForeignKey(-1);
}
}
}說明:拆分為若干棵樹后設置每條數(shù)據(jù)的虛擬id為自己的索引,每棵樹的id互相隔離,
根據(jù)表格數(shù)據(jù)規(guī)律得出子節(jié)點只可能存在于自己節(jié)點以下,以及下一個相同節(jié)地以上,根據(jù)這個規(guī)律設置每個節(jié)點的父id
遞歸封裝為樹結構
代碼如下(示例):
/**
* 遞歸遍歷為樹形結構
*
* @param vo 當前處理的元素
* @param segmentedDatum 每棵樹的數(shù)據(jù)集
*/
private void treeData(MatMaterialBomImportVo vo, List<MatMaterialBomImportVo> segmentedDatum) {
for (int i = vo.getPrimaryKey(); i < segmentedDatum.size(); i++) {
if (i + 1 == segmentedDatum.size()) {
if (vo.getForeignKey() == null) {
getParentId(vo, segmentedDatum);
}
break;
}
int v = Integer.parseInt(vo.getHierarchy());
int vs = Integer.parseInt(segmentedDatum.get(i + 1).getHierarchy());
if (vs == v + 1) {
if (v > 1) {
vo.setForeignKey(segmentedDatum.get(i).getPrimaryKey());
for (int j = vo.getPrimaryKey(); j > 0; j--) {
if (Integer.parseInt(segmentedDatum.get(j).getHierarchy()) == Integer.parseInt(vo.getHierarchy()) - 1) {
vo.setForeignKey(segmentedDatum.get(j).getPrimaryKey());
}
}
}
vo.getImportVoList().add(segmentedDatum.get(i + 1));
}
if (vs <= v) {
if (vo.getForeignKey() == null) {
for (int j = vo.getPrimaryKey(); j > 0; j--) {
if (Integer.parseInt(segmentedDatum.get(j).getHierarchy()) == Integer.parseInt(vo.getHierarchy()) - 1) {
vo.setForeignKey(segmentedDatum.get(j).getPrimaryKey());
break;
}
}
}
break;
}
}
if (vo.getImportVoList() != null && vo.getImportVoList().size() > 0) {
for (MatMaterialBomImportVo matMaterialBomImportVo : vo.getImportVoList()) {
treeData(matMaterialBomImportVo, segmentedDatum);
}
}
}說明:我這里傳進來的vo是沒有設置id和父id的,只對數(shù)據(jù)源做了樹拆分處理,因為業(yè)務需求,后面并沒有使用這套遞歸的方法組裝為樹,所以遞歸代碼可能有點誤差,僅供參考
總結
這里主要針對導入數(shù)據(jù)為樹形,以及沒有具體的id以及父id的處理,在拆分開沒棵樹的數(shù)據(jù)并且每棵樹的節(jié)點有了父子關系后就可以通過正常的流程處理
到此這篇關于Java詳解如何將excel數(shù)據(jù)轉為樹形的文章就介紹到這了,更多相關Java excel內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
JavaWeb實現(xiàn)用戶登錄注冊功能實例代碼(基于Servlet+JSP+JavaBean模式)
這篇文章主要基于Servlet+JSP+JavaBean開發(fā)模式實現(xiàn)JavaWeb用戶登錄注冊功能實例代碼,非常實用,本文介紹的非常詳細,具有參考借鑒價值,感興趣的朋友一起看看吧2016-05-05
java通過JFrame做一個登錄系統(tǒng)的界面完整代碼示例
這篇文章主要介紹了java通過JFrame做一個登錄系統(tǒng)的界面完整代碼示例,具有一定借鑒價值,需要的朋友可以參考下。2017-12-12
Mybatis從3.4.0版本到3.5.7版本的迭代方法實現(xiàn)
本文主要介紹了Mybatis從3.4.0版本到3.5.7版本的迭代方法實現(xiàn),包括主要的功能增強、不兼容的更改和修復的錯誤,具有一定的參考價值,感興趣的可以了解一下2025-03-03
springboot如何通過不同的策略動態(tài)調(diào)用不同的實現(xiàn)類
這篇文章主要介紹了springboot如何通過不同的策略動態(tài)調(diào)用不同的實現(xiàn)類,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02
SpringBoot集成Caffeine緩存的實現(xiàn)步驟
Caffeine cache是一個針對Java的高性能緩存庫。在本文中,我們將介紹它與Spring Boot如何一起使用。2021-05-05
一文教會Java新手使用Spring?MVC中的查詢字符串和查詢參數(shù)
在使用springMVC框架構建web應用,客戶端常會請求字符串、整型、json等格式的數(shù)據(jù),這篇文章主要給大家介紹了關于通過一文教會Java新手使用Spring?MVC中的查詢字符串和查詢參數(shù)的相關資料,需要的朋友可以參考下2024-01-01
Mybatis-Plus支持GBase8s分頁查詢的實現(xiàn)示例
本文主要介紹了使?Mybatis-Plus?支持?GBase8s?的分頁查詢,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-01-01

