JAVA如何把數(shù)據(jù)庫的數(shù)據(jù)處理成樹形結(jié)構(gòu)
前言
不知道大家在做項目的時候有沒有接觸到將平平無奇數(shù)據(jù)結(jié)合處理成有層次的數(shù)據(jù)呢,類似下面這樣
或者 生活處處都有,我想大家都應該接觸過的,下面直接看怎么實現(xiàn),我會大概講一下思路,當然也可以直接跳到最后去看代碼實現(xiàn)的哈
follow me!go go go!
❗此篇文章也只是一個簡單的學習記錄,不詳細的對代碼進行講解
😎實現(xiàn)思路😎
首先一般數(shù)據(jù)庫的模型設計如下
sql腳本
-- ---------------------------- -- Table structure for product -- ---------------------------- DROP TABLE IF EXISTS `product`; CREATE TABLE `product` ( `id` bigint(20) NOT NULL AUTO_INCREMENT, `uuid` varchar(64) NOT NULL, `name` varchar(100) NOT NULL COMMENT '名稱', `sort` int(11) DEFAULT NULL COMMENT '排序', `parent_uuid` varchar(64) NOT NULL DEFAULT '-1' COMMENT '父親 無父級為-1', `level` varchar(10) NOT NULL COMMENT '產(chǎn)品層級', `create_time` datetime NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8mb4 COMMENT='產(chǎn)品表'; -- ---------------------------- -- Records of product -- ---------------------------- INSERT INTO `product` VALUES ('1', '4dbf40d2-2af7-425c-a103-0349caaa26cf', '生產(chǎn)類', '1', '-1', '1', '2021-09-23 15:34:36'); INSERT INTO `product` VALUES ('2', '3062deff-8ec7-44c4-bd4e-88fe3c7b835c', '22', '1', '4dbf40d2-2af7-425c-a103-0349caaa26cf', '2', '2021-09-23 15:37:20'); INSERT INTO `product` VALUES ('3', '32afe426-9337-41c1-83e8-caf3248ba57e', '互聯(lián)網(wǎng)信息', '2', '4dbf40d2-2af7-425c-a103-0349caaa26cf', '2', '2021-09-23 15:38:19'); INSERT INTO `product` VALUES ('4', '34c5239f-db2d-4394-b367-a57f8ae6f8ff', '33', '1', '3062deff-8ec7-44c4-bd4e-88fe3c7b835c', '3', '2021-09-23 15:53:29'); INSERT INTO `product` VALUES ('5', '19eedcd3-aa7f-4a2d-8182-d3f795e99b9d', '44', '1', '34c5239f-db2d-4394-b367-a57f8ae6f8ff', '4', '2021-09-23 15:53:56');
我們觀察一下,可以發(fā)現(xiàn)我們的關注重點在name、uuid、parent_uuid上面:
- name:分類名稱
- uuid:UUID 是 通用唯一識別碼(Universally Unique Identifier)的縮寫,是一種軟件建構(gòu)的標準,其目的,是讓分布式系統(tǒng)中的所有元素,都能有唯一的辨識信息,而不需要通過中央控制端來做辨識信息的指定。這里可以簡單看作一個唯一標識碼(類似于ID但不等于ID)
- parent_uuid:子類的父類UUID,最高級規(guī)定為-1(這個可以自己定義,不會有相同的就好)
下面就是我創(chuàng)建的模擬數(shù)據(jù)
想要實現(xiàn)數(shù)形狀結(jié)構(gòu),肯定要以某一屬性來作為突破口,它就是parent_uuid,那么到底是如何實現(xiàn)的 來看具體代碼
完整代碼
只貼重點代碼
首先使用了Mabatis-generator生成了通用后端代碼,結(jié)構(gòu)如下:
ProductController.class
package com.csdn.caicai.test.modules.product.controller; import io.swagger.annotations.Api; import io.swagger.annotations.ApiOperation; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.*; import com.csdn.caicai.test.modules.product.dto.ProductRsp; import com.csdn.caicai.test.modules.product.biz.IProductBiz; import java.util.List; /** * 產(chǎn)品表 * * @author * @date */ @RestController @Api(tags = {"產(chǎn)品表"}) @RequestMapping("/caicai/product") @Validated public class ProductController { private static final Logger log = LoggerFactory.getLogger(ProductController.class); @Autowired private IProductBiz productBiz; /** * 產(chǎn)品樹 */ @ApiOperation(value = "產(chǎn)品樹") @RequestMapping(path = "/tree", method = RequestMethod.GET) public List<ProductRsp> tree() { return productBiz.tree(); } }
IProductBiz.class
package com.csdn.caicai.test.modules.product.biz; import com.csdn.caicai.test.modules.product.dto.ProductRsp; import java.util.List; /** * @author * @date */ public interface IProductBiz { List<ProductRsp> tree(); }
ProductBiz.class
package com.csdn.caicai.test.modules.product.biz; import org.apache.commons.lang3.StringUtils; import org.assertj.core.util.Lists; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.util.CollectionUtils; import java.util.List; import java.util.stream.Collectors; import tk.mybatis.mapper.entity.Example; import com.csdn.caicai.test.modules.product.service.IProductService; import com.csdn.caicai.test.modules.product.dao.entity.ProductEntity; import com.csdn.caicai.test.modules.product.dto.ProductReq; import com.csdn.caicai.test.modules.product.dto.ProductRsp; import static java.util.stream.Collectors.toList; /** * @author * @date */ @Service("productBiz") public class ProductBiz implements IProductBiz { @Autowired private IProductService productService; /** * 根據(jù)條件查詢 * * @param productReq * @return */ public List<ProductEntity> selectByCondition(ProductReq productReq) { Example example = new Example(ProductEntity.class); //下面添加自定義收索條件 return productService.selectByExample(example); } @Override public List<ProductRsp> tree() { ProductReq req = new ProductReq(); List<ProductRsp> list = selectByCondition(req).stream().map(this::productConvert).collect(Collectors.toList()); return buildTree(list, req.getParentUuid()); } private ProductRsp productConvert(ProductEntity e) { ProductRsp orgNode = new ProductRsp(); orgNode.setId(e.getId()); orgNode.setUuid(e.getUuid()); orgNode.setName(e.getName()); orgNode.setLevel(e.getLevel()); orgNode.setSort(e.getSort()); orgNode.setParentUuid(e.getParentUuid()); return orgNode; } public static List<ProductRsp> buildTree(List<ProductRsp> all, String parentUuid) { if (CollectionUtils.isEmpty(all)) return Lists.newArrayList(); List<ProductRsp> parentList = all.stream() .filter(e -> StringUtils.isBlank(e.getParentUuid()) || "-1".equals(e.getParentUuid()) || e.getParentUuid().equals(parentUuid)) .collect(toList()); getSubList(parentList, all); return parentList; } private static void getSubList(List<ProductRsp> parentList, List<ProductRsp> all) { parentList.forEach(e -> { List<ProductRsp> subList = all.stream().filter(o -> o.getParentUuid().equals(e.getUuid())).collect(toList()); e.setSubList(subList); if (!CollectionUtils.isEmpty(subList)) getSubList(subList, all); }); } }
ProductReq.class
package com.csdn.caicai.test.modules.product.dto; import io.swagger.annotations.ApiModel; import io.swagger.annotations.ApiModelProperty; import lombok.Data; import java.io.Serializable; /** * @author * @date */ @ApiModel(value = "ProductReq", description = "產(chǎn)品表") @Data public class ProductReq implements Serializable { private static final long serialVersionUID = 1L; /** * */ @ApiModelProperty(value = "", name = "id") private Long id; /** * */ @ApiModelProperty(value = "", name = "uuid") private String uuid; /** * 名稱 */ @ApiModelProperty(value = "名稱", name = "name") private String name; /** * 排序 */ @ApiModelProperty(value = "排序", name = "sort") private Integer sort; /** * 父親 無父級為-1 */ @ApiModelProperty(value = "父親 無父級為-1", name = "parentUuid") private String parentUuid; /** * 產(chǎn)品層級 */ @ApiModelProperty(value = "產(chǎn)品層級", name = "level") private String level; }
ProductRsp.class
package com.csdn.caicai.test.modules.product.dto; import io.swagger.annotations.ApiModel; import io.swagger.annotations.ApiModelProperty; import lombok.Data; import java.io.Serializable; import java.util.Date; import java.util.List; /** * @author * @date */ @ApiModel(value = "ProductRsp", description = "產(chǎn)品表") @Data public class ProductRsp implements Serializable { private static final long serialVersionUID = 1L; /** * */ @ApiModelProperty(value = "", name = "id") private Long id; /** * */ @ApiModelProperty(value = "", name = "uuid") private String uuid; /** * 名稱 */ @ApiModelProperty(value = "名稱", name = "name") private String name; /** * 排序 */ @ApiModelProperty(value = "排序", name = "sort") private Integer sort; /** * 父親 無父級為-1 */ @ApiModelProperty(value = "父親 無父級為-1", name = "parentUuid") private String parentUuid; /** * 產(chǎn)品層級 */ @ApiModelProperty(value = "產(chǎn)品層級", name = "level") private String level; /** * */ @ApiModelProperty(value = "", name = "createTime") private Date createTime; @ApiModelProperty(value = "下屬產(chǎn)品", name = "subList") private List<ProductRsp> subList; }
測試一下
可以看到,實現(xiàn)了我們的效果
總結(jié)-核心代碼
上面羅里吧嗦,其實核心代碼就是以下代碼,親們來試著理解一下,然后就可以在此基礎上美化一下就好了:
ProductRsp、ProductReq 是實體類,可以自行替換里面的內(nèi)容
private ProductRsp productConvert(ProductEntity e) { ProductRsp orgNode = new ProductRsp(); orgNode.setId(e.getId()); orgNode.setUuid(e.getUuid()); orgNode.setName(e.getName()); orgNode.setLevel(e.getLevel()); orgNode.setSort(e.getSort()); orgNode.setParentUuid(e.getParentUuid()); return orgNode; } public static List<ProductRsp> buildTree(List<ProductRsp> all, String parentUuid) { if (CollectionUtils.isEmpty(all)) return Lists.newArrayList(); List<ProductRsp> parentList = all.stream() .filter(e -> StringUtils.isBlank(e.getParentUuid()) || "-1".equals(e.getParentUuid()) || e.getParentUuid().equals(parentUuid)) .collect(toList()); getSubList(parentList, all); return parentList; } private static void getSubList(List<ProductRsp> parentList, List<ProductRsp> all) { parentList.forEach(e -> { List<ProductRsp> subList = all.stream().filter(o -> o.getParentUuid().equals(e.getUuid())).collect(toList()); e.setSubList(subList); if (!CollectionUtils.isEmpty(subList)) getSubList(subList, all); }); }
到此這篇關于JAVA如何把數(shù)據(jù)庫的數(shù)據(jù)處理成樹形結(jié)構(gòu)的文章就介紹到這了,更多相關JAVA如何把數(shù)據(jù)庫的數(shù)據(jù)處理成樹形結(jié)構(gòu)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
解析Java?中for循環(huán)和foreach循環(huán)哪個更快
這篇文章主要介紹了Java中for循環(huán)和foreach循環(huán)哪個更快示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09java開發(fā)CPU流水線與指令亂序執(zhí)行詳解
這篇文章主要為大家介紹了java開發(fā)CPU流水線與指令亂序執(zhí)行詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-09-09Java?深入學習static關鍵字和靜態(tài)屬性及方法
這篇文章主要介紹了Java?深入學習static關鍵字和靜態(tài)屬性及方法,文章通過圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-09-09將Swagger2文檔導出為HTML或markdown等格式離線閱讀解析
這篇文章主要介紹了將Swagger2文檔導出為HTML或markdown等格式離線閱讀,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下2019-11-11