溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Element的el-tree控件后臺數(shù)據(jù)結(jié)構(gòu)的生成以及方法的抽取

發(fā)布時(shí)間:2020-09-29 22:03:38 來源:腳本之家 閱讀:582 作者:java_xxxx 欄目:web開發(fā)

最近用到了el-tree控件,主要是數(shù)據(jù)的格式,按照官網(wǎng)的數(shù)據(jù)格式來就可以顯示節(jié)點(diǎn)的樹形結(jié)構(gòu)了。
代碼參考很多

這里給出一個(gè)比較好的鏈接:https://www.jb51.net/article/181990.htm

代碼說明在注釋里寫的很詳細(xì)了已經(jīng),這里不再敘述說明。至于為什么抽取成這種格式的數(shù)據(jù),那是因?yàn)镋lementUI-tree規(guī)定的數(shù)據(jù)格式,你想要用這個(gè)控件,就必須按照他們規(guī)定的這個(gè)格式 來。
數(shù)據(jù)格式如下:

Element的el-tree控件后臺數(shù)據(jù)結(jié)構(gòu)的生成以及方法的抽取

Controller代碼

@RequestMapping("/cateList")
  @ResponseBody
  public List<TbCategory> cateList() {

    // 整體思路:
    // 1、取得所有數(shù)據(jù)、放入集合List1 (tbCategories)
    // 2、將List1所有數(shù)據(jù)都放入到map(treeMap)中:元素id為鍵,元素本身對象為值
    // 3、取得頂層節(jié)點(diǎn)放入集合List2中(resultList)
    // 4、遍歷List1中的所有數(shù)據(jù),通過數(shù)據(jù)的parentId為鍵在map中取值
    //   1)如果能取到,則說明該元素有父節(jié)點(diǎn)
    //      1、判斷該父節(jié)點(diǎn)下的childList中是否有已經(jīng)子節(jié)點(diǎn)
    //       1、若無:則創(chuàng)建一個(gè)集合,將子節(jié)點(diǎn)放入
    //       2、若有:則直接將子節(jié)點(diǎn)放入即可
    // 5、把放好的數(shù)據(jù)放回到map中
    // 6、返回List2(resultList)

    // 注意:整個(gè)過程將所有數(shù)據(jù)取出放入list2(resultList),返回的也是  //list2


    List<TbCategory> tbCategories = categoryService.cateList();

    List<TbCategory> resultList = new ArrayList<TbCategory>(); // 存貯頂層的數(shù)據(jù)

    Map<Object ,Object> treeMap = new HashMap();
    Object itemTree;

    for(int i = 0;i<tbCategories.size() && !tbCategories.isEmpty();i++){
      itemTree = tbCategories.get(i);
      treeMap.put(tbCategories.get(i).getNodeId(),tbCategories.get(i));// 把所有的數(shù)據(jù)都放到map中


    }


// 這里也可以用另一種方法,就是拿到集合里的每個(gè)元素的父id去數(shù)據(jù)庫中查詢,但是,這樣與數(shù)據(jù)庫的交互次數(shù)就太多了
 // 遍歷map得到頂層節(jié)點(diǎn)(游離節(jié)點(diǎn)也算作頂層節(jié)點(diǎn))
    for(int i =0;i<tbCategories.size();i++){
      // 優(yōu)點(diǎn)1:整個(gè)方法,只查詢了一次數(shù)據(jù)庫
      // 優(yōu)點(diǎn)2:不用知道頂層節(jié)點(diǎn)的id
     if(!treeMap.containsKey(tbCategories.get(i).getParentId())){
        // 我們在存儲的時(shí)候就是將元素的id為鍵,元素本身為值存入的
        // 以元素的父id為鍵,在map里取值,若取不到則,對應(yīng)的元素不存在,即沒有父節(jié)點(diǎn),為頂層節(jié)點(diǎn)或游離節(jié)點(diǎn)
        // 將頂層節(jié)點(diǎn)放入list集合
        resultList.add(tbCategories.get(i));
      }
    }

    // 循環(huán)數(shù)據(jù),將數(shù)據(jù)放到該節(jié)點(diǎn)的父節(jié)點(diǎn)的children屬性中
    for(int i =0 ;i<tbCategories.size()&& !tbCategories.isEmpty();i++){
      // 數(shù)據(jù)庫中,若一個(gè)元素有子節(jié)點(diǎn),那么,該元素的id為子節(jié)點(diǎn)的父id
      //treeMap.get(tbCategories.get(i).getParentId()); // 從map集合中找到父節(jié)點(diǎn)
      TbCategory category = (TbCategory)treeMap.get(tbCategories.get(i).getParentId());
      if(category!=null ){ // 不等于null,也就意味著有父節(jié)點(diǎn)
        // 有了父節(jié)點(diǎn),要判斷父節(jié)點(diǎn)下存貯字節(jié)點(diǎn)的集合是否存在,然后將子節(jié)點(diǎn)放入
        if(category.getChildList() == null){
          // 判斷一個(gè)集合是否被創(chuàng)建用null:表示結(jié)合還沒有被分配內(nèi)存空間(即還沒有被創(chuàng)建),內(nèi)存大小自然為null
          // 用集合的size判斷集合中是否有元素,為0,沒有元素(集合已經(jīng)被創(chuàng)建),
          category.setChildList(new ArrayList<TbCategory>());
        }
        category.getChildList().add(tbCategories.get(i)); // 添加到父節(jié)點(diǎn)的ChildList集合下

        // 這一步其實(shí)可以不要,因?yàn)槲覀冃薷牧藬?shù)據(jù)(添加了子節(jié)點(diǎn),然后在將元素放入到map中,
        // 若鍵相同,map會自動覆蓋掉相同的鍵值對,達(dá)到更新map集合中的數(shù)據(jù)的目的),但是我們
        // 這里只是從map中取值,而并不關(guān)心值的子節(jié)點(diǎn)(子節(jié)點(diǎn)是對象本身自己封裝的。這里我們知道
        // 元素從查詢后放入map,父節(jié)點(diǎn)放入list,然后通過鍵來在map中取得對象,之后再將修改過的對象重新放入map當(dāng)中
        // ,我們并沒有直接操作list,但是在list中對象的值卻是已經(jīng)修改過了,這就是對象的引用傳遞,同一個(gè)引用對象是通過
        // 地址值來操作對象的,即有不同的引用,但是對象中的屬性是已經(jīng)通過引用的操作而改變的,所以這里一旦修改過后,無論是map中還是list中,再次取值時(shí)都已經(jīng)是更改過后的值了)
        treeMap.put(tbCategories.get(i).getParentId(),category); // 把放好的數(shù)據(jù)放回到map中
      }

    }


   return resultList;
  }

實(shí)體類:

 private Long nodeId;
  private String categoryName;
  private Long parentId;
  private Long childId;
  private List<T> childList;

以上數(shù)據(jù)都在后臺封裝好了,前臺直接獲取數(shù)據(jù)顯示即可

<el-tree :data="treeList"
       :props="defaultProps"
       @node-click="handleNodeClick"
       node-key="nodeId"
       show-checkbox=true>
  </el-tree>

js:

defaultProps:{
        children: 'childList',
        label: 'categoryName' // 這里的名字要和你封裝的數(shù)據(jù)中的節(jié)點(diǎn)的名字一樣
      }
// 點(diǎn)擊事件
 handleNodeClick: function (data) {
        console.log("沒做處理");
      }

方法抽取

上面的方法雖然也能用,但是想把這個(gè)方法抽取成一個(gè)通用的方法,以后再寫的時(shí)候就可以直接調(diào)取該方法了。抽取的過程中還是遇到了很多的問題的。

例如實(shí)體類A是對應(yīng)數(shù)據(jù)庫中存儲節(jié)點(diǎn)的表的實(shí)體類,但是,實(shí)體類中是不存在setChildList、getChildList集合這些方法的。我就把這些存貯信息的字段寫在了一個(gè)工具類里面,然后方法也在該工具類里。這樣返回的時(shí)候就要返回一個(gè)裝有該工具類的一個(gè)集合了。還有一個(gè)問題就是該工具類中都需要那些字段?因?yàn)槭且豢脴?,所以我們需要?jié)點(diǎn)id,幾點(diǎn)的父id,節(jié)點(diǎn)的名稱,以及存儲子節(jié)點(diǎn)的集合,如果想要更多的數(shù)據(jù),可以添加一個(gè) T data泛型,該泛型直接將原本存儲節(jié)點(diǎn)的實(shí)體類對象存儲進(jìn)來。這樣所有的數(shù)據(jù)就都整齊了,數(shù)據(jù)結(jié)構(gòu)也就完整了。

contrell代碼:

因?yàn)槲覀兿氤槿∫粋€(gè)公用的方法,那么參數(shù)的類型就是不確定的,即傳入的list中元素的類型是不固定的,所以要用到泛型。上面說過了,我們要拿到節(jié)點(diǎn)id,父id,以及節(jié)點(diǎn)的名稱賦值給工具類中對應(yīng)的字段。但是既然是用泛型,所以就不知道對象類型,所以我們在工具類中是點(diǎn)不出相應(yīng)的方法來取到值的,我們就要用到反射,反射的知道類中具體的字段或者方法的名字來取值,所以我們在controller來調(diào)用工具類的時(shí)候就要,將節(jié)點(diǎn)id、父id和節(jié)點(diǎn)名稱傳入。當(dāng)然了還要傳入從后臺查詢到了裝有數(shù)據(jù)的list集合。

contreller代碼如下所示:

@RequestMapping("/cateList")
@ResponseBody
public List<TreeUtils> cateList() throws Exception{
    List<TbCategory> tbCategories = categoryService.cateList();
    List<TreeUtils> treeList = TreeUtils.getTreeList(tbCategories,"nodeId", "parentId", "categoryName");

    return treeList;
  }

工具類:

// 抽取方法的時(shí)候要考慮一個(gè)問題,即,返回一個(gè)集合,集合中有父節(jié)點(diǎn)和字節(jié)點(diǎn),父節(jié)點(diǎn)和字節(jié)點(diǎn)的類型一定要統(tǒng)一,
// 即這里返回的是一個(gè)裝有TreeUtils類型的集合,那么集合里的父節(jié)點(diǎn)和子節(jié)點(diǎn)一定都得是TreeUtils類型的

public class TreeUtils<T> {

  private Integer id;      // 節(jié)點(diǎn)id
  private Integer parentId;  // 父節(jié)點(diǎn)
  private String name;     // 節(jié)點(diǎn)名稱 ,返回給前臺的是一個(gè)裝有TreeUtils的集合的數(shù)據(jù),所以在前臺顯示數(shù)據(jù)的時(shí)候,el-tree的lable的名字的和這個(gè)一樣
  private List<TreeUtils> childList; // 父節(jié)點(diǎn)中存放子節(jié)點(diǎn)的集合
  private T data;       // 節(jié)點(diǎn)數(shù)據(jù)

方法

/**
   * @param listData  // 從數(shù)據(jù)庫中查詢的數(shù)據(jù)
   * @return
   */
  public static List<TreeUtils> getTreeList(List<?> listData ,String id,String parentId,String categoryName) throws Exception{

    List<TreeUtils> resultList = new ArrayList<TreeUtils>(); // 最終返回的結(jié)果
    Map<Integer ,Object> map = new HashMap<Integer,Object>();

    for(int i =0;i<listData.size() && !listData.isEmpty();i++){

      // 寫一個(gè)與該方法差不多的方法,將得到TreeUtils的代碼抽取出來
      // 也可以將listData集合整個(gè)轉(zhuǎn)換成裝有TreeUtils的集合x,然后再循環(huán)x
      TreeUtils treeUtils = new TreeUtils();
      treeUtils.setId(Integer.parseInt(TreeUtils.getFileValue(listData.get(i),id).toString())); // id       // 返回值為Object無法直接轉(zhuǎn)換成Integer,先toString,再轉(zhuǎn)換成Integer。這里的返回值寫成Object是因?yàn)槎喾N類型字段的值都可以用該方法
      treeUtils.setParentId(Integer.parseInt(TreeUtils.getFileValue(listData.get(i),parentId).toString())); // 父id
      treeUtils.setName(TreeUtils.getFileValue(listData.get(i),categoryName).toString()); // 節(jié)點(diǎn)名
      //System.out.println("節(jié)點(diǎn)名為+"+TreeUtils.getFileValue(listData.get(i),categoryName).toString());
      treeUtils.setData(listData.get(i));  // data:原對象中的所有屬性,無children

      // 通過反射得到每條數(shù)據(jù)的id將數(shù)據(jù)封裝的map集合中,id為鍵,元素本身為值
      map.put(treeUtils.getId(),treeUtils);


      // 將所有頂層元素添加到resultList集合中
      //if( 0 == treeUtils.getParentId()){
       //  resultList.add(treeUtils);
      // }
    }
// 得到所有的頂層節(jié)點(diǎn),游離節(jié)點(diǎn)也算作頂層節(jié)點(diǎn)
// 優(yōu)點(diǎn)一,不用知道等級節(jié)點(diǎn)的id
// 優(yōu)點(diǎn)而,只查詢了一次數(shù)據(jù)庫
    for(int i =0;i<listData.size();i++){
      if(!map.containsKey(Integer.parseInt(TreeUtils.getFileValue(listData.get(i),parentId).toString()))){
        resultList.add((TreeUtils) map.get(Integer.parseInt(TreeUtils.getFileValue(listData.get(i),id).toString())));
      }
    }



    for(int i =0;i<listData.size() && !listData.isEmpty();i++){
      TreeUtils obj = (TreeUtils)map.get(Integer.parseInt(TreeUtils.getFileValue(listData.get(i), parentId).toString()));
      if(obj != null){
        if(obj.getChildList() == null){
          obj.setChildList(new ArrayList());
        }
        obj.getChildList().add(map.get(Integer.parseInt(TreeUtils.getFileValue(listData.get(i),id).toString())));
      }
    }
    return resultList;
  }

反射的方法

/**
   * 通過反射得到的數(shù)據(jù)類型的也是不一定的,所以這里我們返回值為Object
   * Object是無法直接轉(zhuǎn)為Integer,現(xiàn)將Object轉(zhuǎn)為String,然后再將String轉(zhuǎn)為Integer
   * @param item
   * @param fileName
   * @return
   */
  public static Object getFileValue(Object item,String fileName) throws Exception {
    Class<?> aClass = item.getClass();
    Field file = aClass.getDeclaredField(fileName); // 得到所有字段包括私有字段
    file.setAccessible(true); // 取消訪問限制
    return file.get(item);  // 這里就體現(xiàn)出反射的意思了,我們通常都是通過對象拿到字段,這里是通過字段,將類的字節(jié)碼對象為參數(shù)傳入,來得到值
  }

ps:抽取方法遇到了很多的問題,其中的T,?等泛型還只是會簡單的用,并不熟練,以后要多加學(xué)習(xí)。

 到此這篇關(guān)于Element的el-tree控件后臺數(shù)據(jù)結(jié)構(gòu)的生成以及方法的抽取的文章就介紹到這了,更多相關(guān)Element el-tree生成及方法抽取內(nèi)容請搜索億速云以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持億速云!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI