Java詳解如何將excel數(shù)據轉為樹形
前言
今天收到一個導入的任務,要求將excel數(shù)據保存到數(shù)據庫中,不同于普通的導入,這個導入的數(shù)據是一個樹形結構,如下圖:
通過觀察數(shù)據中的層級列我們發(fā)現(xiàn)表格數(shù)據由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ù)據
1.創(chuàng)建實體類
創(chuàng)建vo接收解析數(shù)據,在這里,我們只關心層級屬性
@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ù)據
將數(shù)據源拆分為若干棵樹的數(shù)據集
代碼如下(示例):
/** * 將集合對象按指定元素分割存儲 * * @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ù)據源 for (; count < materialVos.size(); count++) { //第一個必然為樹的根節(jié)點,直接獲取并跳過 if (count == 0) { bomImportVo.add(materialVos.get(count)); continue; } //當數(shù)據源第n個等于根節(jié)點并且已經成功添加過數(shù)據時判斷為一段數(shù)據的結束,跳出循環(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ù)據的虛擬id為自己的索引,每棵樹的id互相隔離,
根據表格數(shù)據規(guī)律得出子節(jié)點只可能存在于自己節(jié)點以下,以及下一個相同節(jié)地以上,根據這個規(guī)律設置每個節(jié)點的父id
遞歸封裝為樹結構
代碼如下(示例):
/** * 遞歸遍歷為樹形結構 * * @param vo 當前處理的元素 * @param segmentedDatum 每棵樹的數(shù)據集 */ 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ù)據源做了樹拆分處理,因為業(yè)務需求,后面并沒有使用這套遞歸的方法組裝為樹,所以遞歸代碼可能有點誤差,僅供參考
總結
這里主要針對導入數(shù)據為樹形,以及沒有具體的id以及父id的處理,在拆分開沒棵樹的數(shù)據并且每棵樹的節(jié)點有了父子關系后就可以通過正常的流程處理
到此這篇關于Java詳解如何將excel數(shù)據轉為樹形的文章就介紹到這了,更多相關Java excel內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
第一次使用Android Studio時你應該知道的一切配置(推薦)
這篇文章主要介紹了第一次使用Android Studio時你應該知道的一切配置(推薦) ,需要的朋友可以參考下2017-09-09java8中NIO緩沖區(qū)(Buffer)的數(shù)據存儲詳解
在本篇文章中小編給大家分享了關于java8中NIO緩沖區(qū)(Buffer)的數(shù)據存儲的相關知識點,需要的朋友們參考下。2019-04-04Spring?main方法中如何調用Dao層和Service層的方法
這篇文章主要介紹了Spring?main方法中調用Dao層和Service層的方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-12-12java 解決異常 2 字節(jié)的 UTF-8 序列的字節(jié)2 無效的問題
這篇文章主要介紹了java 解決異常 2 字節(jié)的 UTF-8 序列的字節(jié) 2 無效的問題的相關資料,需要的朋友可以參考下2016-12-12