Android 打造任意层级树形控件 考验你的数据结构和设计

简介: Android 打造任意层级树形控件 考验你的数据结构和设计

1、概述
大家在项目中或多或少的可能会见到,偶尔有的项目需要在APP上显示个树形控件,比如展示一个机构组织,最上面是boss,然后各种部门,各种小boss,最后各种小罗罗;整体是一个树形结构;遇到这样的情况,大家可能回去百度,因为层次多嘛,可能更容易想到ExpandableListView , 因为这玩意层级比Listview多,但是ExpandableListView实现目前只支持两级,当然也有人改造成多级的;但是从我个人角度去看,首先我不喜欢ExpandableListView ,数据集的组织比较复杂。所以今天带大家使用ListView来打造一个树形展示效果。ListView应该是大家再熟悉不过的控件了,并且数据集也就是个List 。

本篇博客目标实现,只要是符合树形结构的数据可以轻松的通过我们的代码,实现树形效果,有多轻松,文末就知道了~~

好了,既然是要展现树形结构,那么数据上肯定就是树形的一个依赖,也就是说,你的每条记录,至少有个字段指向它的父节点;类似(id , pId, others ....)

2、原理分析
先看看我们的效果图:

我们支持任意层级,包括item的布局依然让用户自己的去控制,我们的demo的Item布局很简单,一个图标+文本~~

原理就是,树形不树形,其实不就是多个缩进么,只要能够判断每个item属于树的第几层(术语貌似叫高度),设置合适的缩进即可。

当然了,原理说起来简单,还得控制每一层间关系,添加展开缩回等,以及有了缩进还要能显示在正确的位置,不过没关系,我会带着大家一步一步实现的。

3、用法
由于整体比较长,我决定首先带大家看一下用法,就是如果学完了这篇博客,我们需要树形控件,我们需要花多少精力去完成~~

现在需求来了:我现在需要展示一个文件管理系统的树形结构:

数据是这样的:

//id , pid , label , 其他属性
       mDatas.add(new FileBean(1, 0, "文件管理系统"));
       mDatas.add(new FileBean(2, 1, "游戏"));
       mDatas.add(new FileBean(3, 1, "文档"));
       mDatas.add(new FileBean(4, 1, "程序"));
       mDatas.add(new FileBean(5, 2, "war3"));
       mDatas.add(new FileBean(6, 2, "刀塔传奇"));

       mDatas.add(new FileBean(7, 4, "面向对象"));
       mDatas.add(new FileBean(8, 4, "非面向对象"));

       mDatas.add(new FileBean(9, 7, "C++"));
       mDatas.add(new FileBean(10, 7, "JAVA"));
       mDatas.add(new FileBean(11, 7, "Javascript"));
       mDatas.add(new FileBean(12, 8, "C"));

当然了,bean可以有很多属性,我们提供你动态的设置树节点上的显示、以及不约束id, pid 的命名,你可以起任意丧心病狂的属性名称;
那么我们如何确定呢?

看下Bean:

package com.zhy.bean;

import com.zhy.tree.bean.TreeNodeId;
import com.zhy.tree.bean.TreeNodeLabel;
import com.zhy.tree.bean.TreeNodePid;

public class FileBean
{
   @TreeNodeId
   private int _id;
   @TreeNodePid
   private int parentId;
   @TreeNodeLabel
   private String name;
   private long length;
   private String desc;

   public FileBean(int _id, int parentId, String name)
   {
       super();
       this._id = _id;
       this.parentId = parentId;
       this.name = name;
   }

}

现在,不用说,应该也知道我们通过注解来确定的。
下面看我们如何将这数据转化为树

布局文件就一个listview,就补贴了,直接看Activity

package com.zhy.tree_view;

import java.util.ArrayList;
import java.util.List;

import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;

import com.zhy.bean.FileBean;
import com.zhy.tree.bean.TreeListViewAdapter;

public class MainActivity extends Activity
{
   private List mDatas = new ArrayList();
   private ListView mTree;
   private TreeListViewAdapter mAdapter;

   @Override
   protected void onCreate(Bundle savedInstanceState)
   {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_main);

       initDatas();
       mTree = (ListView) findViewById(R.id.id_tree);
       try
       {
           
           mAdapter = new SimpleTreeAdapter(mTree, this, mDatas, 10);
           mTree.setAdapter(mAdapter);
       } catch (IllegalAccessException e)
       {
           e.printStackTrace();
       }

   }

   private void initDatas()
   {

       // id , pid , label , 其他属性
       mDatas.add(new FileBean(1, 0, "文件管理系统"));
       mDatas.add(new FileBean(2, 1, "游戏"));
       mDatas.add(new FileBean(3, 1, "文档"));
       mDatas.add(new FileBean(4, 1, "程序"));
       mDatas.add(new FileBean(5, 2, "war3"));
       mDatas.add(new FileBean(6, 2, "刀塔传奇"));

       mDatas.add(new FileBean(7, 4, "面向对象"));
       mDatas.add(new FileBean(8, 4, "非面向对象"));

       mDatas.add(new FileBean(9, 7, "C++"));
       mDatas.add(new FileBean(10, 7, "JAVA"));
       mDatas.add(new FileBean(11, 7, "Javascript"));
       mDatas.add(new FileBean(12, 8, "C"));

   }

}

Activity里面并没有什么特殊的代码,拿到listview,传入mData,当中初始化了一个Adapter;
看来我们的核心代码都在我们的Adapter里面:

那么看一眼我们的Adapter

package com.zhy.tree_view;

import java.util.List;

import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

import com.zhy.tree.bean.Node;
import com.zhy.tree.bean.TreeListViewAdapter;

public class SimpleTreeAdapter extends TreeListViewAdapter
{

   public SimpleTreeAdapter(ListView mTree, Context context, List datas,
           int defaultExpandLevel) throws IllegalArgumentException,
           IllegalAccessException
   {
       super(mTree, context, datas, defaultExpandLevel);
   }

   @Override
   public View getConvertView(Node node , int position, View convertView, ViewGroup parent)
   {
       
       ViewHolder viewHolder = null;
       if (convertView == null)
       {
           convertView = mInflater.inflate(R.layout.list_item, parent, false);
           viewHolder = new ViewHolder();
           viewHolder.icon = (ImageView) convertView
                   .findViewById(R.id.id_treenode_icon);
           viewHolder.label = (TextView) convertView
                   .findViewById(R.id.id_treenode_label);
           convertView.setTag(viewHolder);

       } else
       {
           viewHolder = (ViewHolder) convertView.getTag();
       }

       if (node.getIcon() == -1)
       {
           viewHolder.icon.setVisibility(View.INVISIBLE);
       } else
       {
           viewHolder.icon.setVisibility(View.VISIBLE);
           viewHolder.icon.setImageResource(node.getIcon());
       }
       viewHolder.label.setText(node.getName());
       
       return convertView;
   }

   private final class ViewHolder
   {
       ImageView icon;
       TextView label;
   }

}

我们的SimpleTreeAdapter继承了我们的TreeListViewAdapter ; 除此之外,代码上只需要复写getConvertView , 且getConvetView其实和我们平时的getView写法一致;
公布出getConvertView 的目的是,让用户自己去决定Item的展示效果。其他的代码,我已经打包成jar了,用的时候导入即可。这样就完成了我们的树形控件。

也就是说用我们的树形控件,只需要将传统继承BaseAdapter改为我们的TreeListViewAdapter ,然后去实现getConvertView 就好了。

那么现在的效果是:

默认就全打开了,因为我们也支持动态设置打开的层级,方面使用者使用。

用起来是不是很随意,加几个注解,ListView的Adapater换个类继承下~~好了,下面开始带大家一起从无到有的实现~

4、实现

1、思路
我们的思路是这样的,我们显示时,需要很多属性,我们需要知道当前节点是否是父节点,当前的层级,他的孩子节点等等;但是用户的数据集是不固定的,最多只能给出类似id,pId 这样的属性。也就是说,用户给的bean并不适合我们用于控制显示,于是我们准备这样做:

1、在用户的Bean中提取出必要的几个元素 id , pId , 以及显示的文本(通过注解+反射);然后组装成我们的真正显示时的Node;即List -> List

2、显示的并非是全部的Node,比如某些节点的父节点是关闭状态,我们需要进行过滤;即List ->过滤后的List

3、显示时,比如点击父节点,它的子节点会跟随其后显示,我们内部是个List,也就是说,这个List的顺序也是很关键的;当然排序我们可以放为步骤一;

最后将过滤后的Node进行显示,设置左内边距即可。

说了这么多,首先看一眼我们封装后的Node

2、Node
package com.zhy.tree.bean;

import java.util.ArrayList;
import java.util.List;

import org.w3c.dom.NamedNodeMap;

import android.util.Log;

public class Node
{

   private int id;
   /**
    * 根节点pId为0
    */
   private int pId = 0;

   private String name;

   /**
    * 当前的级别
    */
   private int level;

   /**
    * 是否展开
    */
   private boolean isExpand = false;

   private int icon;

   /**
    * 下一级的子Node
    */
   private List children = new ArrayList();

   /**
    * 父Node
    */
   private Node parent;

   public Node()
   {
   }

   public Node(int id, int pId, String name)
   {
       super();
       this.id = id;
       this.pId = pId;
       this.name = name;
   }

   public int getIcon()
   {
       return icon;
   }

   public void setIcon(int icon)
   {
       this.icon = icon;
   }

   public int getId()
   {
       return id;
   }

   public void setId(int id)
   {
       this.id = id;
   }

   public int getpId()
   {
       return pId;
   }

   public void setpId(int pId)
   {
       this.pId = pId;
   }

   public String getName()
   {
       return name;
   }

   public void setName(String name)
   {
       this.name = name;
   }

   public void setLevel(int level)
   {
       this.level = level;
   }

   public boolean isExpand()
   {
       return isExpand;
   }

   public List getChildren()
   {
       return children;
   }

   public void setChildren(List children)
   {
       this.children = children;
   }

   public Node getParent()
   {
       return parent;
   }

   public void setParent(Node parent)
   {
       this.parent = parent;
   }

   /**
    * 是否为跟节点
    *
    * @return
    */
   public boolean isRoot()
   {
       return parent == null;
   }

   /**
    * 判断父节点是否展开
    *
    * @return
    */
   public boolean isParentExpand()
   {
       if (parent == null)
           return false;
       return parent.isExpand();
   }

   /**
    * 是否是叶子界点
    *
    * @return
    */
   public boolean isLeaf()
   {
       return children.size() == 0;
   }

   /**
    * 获取level
    */
   public int getLevel()
   {
       return parent == null ? 0 : parent.getLevel() + 1;
   }

   /**
    * 设置展开
    *
    * @param isExpand
    */
   public void setExpand(boolean isExpand)
   {
       this.isExpand = isExpand;
       if (!isExpand)
       {

           for (Node node : children)
           {
               node.setExpand(isExpand);
           }
       }
   }

}

包含了树节点一些常见的属性,一些常见的方法;对于getLevel,setExpand这些方法,大家可以好好看看~

有了Node,刚才的用法中,出现的就是我们Adapter所继承的超类:TreeListViewAdapter;核心代码都在里面,我们准备去一探究竟:

3、TreeListViewAdapter
代码不是很长,直接完整的贴出:

package com.zhy.tree.bean;

import java.util.List;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.BaseAdapter;
import android.widget.ListView;

public abstract class TreeListViewAdapter extends BaseAdapter
{

   protected Context mContext;
   /**
    * 存储所有可见的Node
    */
   protected List mNodes;
   protected LayoutInflater mInflater;
   /**
    * 存储所有的Node
    */
   protected List mAllNodes;

   /**
    * 点击的回调接口
    */
   private OnTreeNodeClickListener onTreeNodeClickListener;

   public interface OnTreeNodeClickListener
   {
       void onClick(Node node, int position);
   }

   public void setOnTreeNodeClickListener(
           OnTreeNodeClickListener onTreeNodeClickListener)
   {
       this.onTreeNodeClickListener = onTreeNodeClickListener;
   }

   /**
    *
    * @param mTree
    * @param context
    * @param datas
    * @param defaultExpandLevel
    *            默认展开几级树
    * @throws IllegalArgumentException
    * @throws IllegalAccessException
    */
   public TreeListViewAdapter(ListView mTree, Context context, List datas,
           int defaultExpandLevel) throws IllegalArgumentException,
           IllegalAccessException
   {
       mContext = context;
       /**
        * 对所有的Node进行排序
        */
       mAllNodes = TreeHelper.getSortedNodes(datas, defaultExpandLevel);
       /**
        * 过滤出可见的Node
        */
       mNodes = TreeHelper.filterVisibleNode(mAllNodes);
       mInflater = LayoutInflater.from(context);

       /**
        * 设置节点点击时,可以展开以及关闭;并且将ItemClick事件继续往外公布
        */
       mTree.setOnItemClickListener(new OnItemClickListener()
       {
           @Override
           public void onItemClick(AdapterView parent, View view,
                   int position, long id)
           {
               expandOrCollapse(position);

               if (onTreeNodeClickListener != null)
               {
                   onTreeNodeClickListener.onClick(mNodes.get(position),
                           position);
               }
           }

       });

   }

   /**
    * 相应ListView的点击事件 展开或关闭某节点
    *
    * @param position
    */
   public void expandOrCollapse(int position)
   {
       Node n = mNodes.get(position);

       if (n != null)// 排除传入参数错误异常
       {
           if (!n.isLeaf())
           {
               n.setExpand(!n.isExpand());
               mNodes = TreeHelper.filterVisibleNode(mAllNodes);
               notifyDataSetChanged();// 刷新视图
           }
       }
   }

   @Override
   public int getCount()
   {
       return mNodes.size();
   }

   @Override
   public Object getItem(int position)
   {
       return mNodes.get(position);
   }

   @Override
   public long getItemId(int position)
   {
       return position;
   }

   @Override
   public View getView(int position, View convertView, ViewGroup parent)
   {
       Node node = mNodes.get(position);
       convertView = getConvertView(node, position, convertView, parent);
       // 设置内边距
       convertView.setPadding(node.getLevel() * 30, 3, 3, 3);
       return convertView;
   }

   public abstract View getConvertView(Node node, int position,
           View convertView, ViewGroup parent);

}

首先我们的类继承自BaseAdapter,然后我们对应的数据集是,过滤出的可见的Node;
我们的构造方法默认接收4个参数:listview,context,mdatas,以及默认展开的级数:0只显示根节点;

可以在构造方法中看到:对用户传入的数据集做了排序,和过滤的操作;一会再看这些方法,这些方法我们使用了一个TreeHelper进行了封装。

注:如果你觉得你的Item布局十分复杂,且布局会展示Bean的其他数据,那么为了方便,你可以让Node中包含一个泛型T , 每个Node携带与之对于的Bean的所有数据;

可以看到我们还直接为Item设置了点击事件,因为我们树,默认就有点击父节点展开与关闭;但是为了让用户依然可用点击监听,我们自定义了一个点击的回调供用户使用;

当用户点击时,默认调用expandOrCollapse方法,将当然节点重置展开标志,然后重新过滤出可见的Node,最后notifyDataSetChanged即可;

其他的方法都是BaseAdapter默认的一些方法了。

下面我们看下TreeHelper中的一些方法:

4、TreeHelper
首先看TreeListViewAdapter构造方法中用到的两个方法:

/**
    * 传入我们的普通bean,转化为我们排序后的Node
    * @param datas
    * @param defaultExpandLevel
    * @return
    * @throws IllegalArgumentException
    * @throws IllegalAccessException
    */
   public static  List getSortedNodes(List datas,
           int defaultExpandLevel) throws IllegalArgumentException,
           IllegalAccessException

   {
       List result = new ArrayList();
       //将用户数据转化为List以及设置Node间关系
       List nodes = convetData2Node(datas);
       //拿到根节点
       List rootNodes = getRootNodes(nodes);
       //排序
       for (Node node : rootNodes)
       {
           addNode(result, node, defaultExpandLevel, 1);
       }
       return result;
   }

拿到用户传入的数据,转化为List以及设置Node间关系,然后根节点,从根往下遍历进行排序;
接下来看:filterVisibleNode

/**
    * 过滤出所有可见的Node
    *
    * @param nodes
    * @return
    */
   public static List filterVisibleNode(List nodes)
   {
       List result = new ArrayList();

       for (Node node : nodes)
       {
           // 如果为跟节点,或者上层目录为展开状态
           if (node.isRoot() || node.isParentExpand())
           {
               setNodeIcon(node);
               result.add(node);
           }
       }
       return result;
   }

过滤Node的代码很简单,遍历所有的Node,只要是根节点或者父节点是展开状态就添加返回;
最后看看这两个方法用到的别的一些私有方法:

   /**
    * 将我们的数据转化为树的节点
    *
    * @param datas
    * @return
    * @throws NoSuchFieldException
    * @throws IllegalAccessException
    * @throws IllegalArgumentException
    */
   private static  List convetData2Node(List datas)
           throws IllegalArgumentException, IllegalAccessException

   {
       List nodes = new ArrayList();
       Node node = null;

       for (T t : datas)
       {
           int id = -1;
           int pId = -1;
           String label = null;
           Class clazz = t.getClass();
           Field[] declaredFields = clazz.getDeclaredFields();
           for (Field f : declaredFields)
           {
               if (f.getAnnotation(TreeNodeId.class) != null)
               {
                   f.setAccessible(true);
                   id = f.getInt(t);
               }
               if (f.getAnnotation(TreeNodePid.class) != null)
               {
                   f.setAccessible(true);
                   pId = f.getInt(t);
               }
               if (f.getAnnotation(TreeNodeLabel.class) != null)
               {
                   f.setAccessible(true);
                   label = (String) f.get(t);
               }
               if (id != -1 && pId != -1 && label != null)
               {
                   break;
               }
           }
           node = new Node(id, pId, label);
           nodes.add(node);
       }

       /**
        * 设置Node间,父子关系;让每两个节点都比较一次,即可设置其中的关系
        */
       for (int i = 0; i < nodes.size(); i++)
       {
           Node n = nodes.get(i);
           for (int j = i + 1; j < nodes.size(); j++)
           {
               Node m = nodes.get(j);
               if (m.getpId() == n.getId())
               {
                   n.getChildren().add(m);
                   m.setParent(n);
               } else if (m.getId() == n.getpId())
               {
                   m.getChildren().add(n);
                   n.setParent(m);
               }
           }
       }

       // 设置图片
       for (Node n : nodes)
       {
           setNodeIcon(n);
       }
       return nodes;
   }

   private static List getRootNodes(List nodes)
   {
       List root = new ArrayList();
       for (Node node : nodes)
       {
           if (node.isRoot())
               root.add(node);
       }
       return root;
   }

   /**
    * 把一个节点上的所有的内容都挂上去
    */
   private static void addNode(List nodes, Node node,
           int defaultExpandLeval, int currentLevel)
   {

       nodes.add(node);
       if (defaultExpandLeval >= currentLevel)
       {
           node.setExpand(true);
       }

       if (node.isLeaf())
           return;
       for (int i = 0; i < node.getChildren().size(); i++)
       {
           addNode(nodes, node.getChildren().get(i), defaultExpandLeval,
                   currentLevel + 1);
       }
   }

   /**
    * 设置节点的图标
    *
    * @param node
    */
   private static void setNodeIcon(Node node)
   {
       if (node.getChildren().size() > 0 && node.isExpand())
       {
           node.setIcon(R.drawable.tree_ex);
       } else if (node.getChildren().size() > 0 && !node.isExpand())
       {
           node.setIcon(R.drawable.tree_ec);
       } else
           node.setIcon(-1);

   }

convetData2Node即遍历用户传入的Bean,转化为Node,其中Id,pId,label通过注解加反射获取;然后设置Node间关系;
getRootNodes 这个简单,获得根节点

addNode :通过递归的方式,把一个节点上的所有的子节点等都按顺序放入;

setNodeIcon :设置图标,这里标明,我们的jar还依赖两个小图标,即两个三角形;如果你觉得树不需要这样的图标,可以去掉;

5、注解的类
最后就是我们的3个注解类了,没撒用,就启到一个标识的作用

TreeNodeId

package com.zhy.tree.bean;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TreeNodeId
{
}

TreeNodePid

package com.zhy.tree.bean;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TreeNodePid
{

}
TreeNodeLabel
package com.zhy.tree.bean;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TreeNodeLabel
{

}

5、最后的展望
基于上面的例子,我们还有很多地方可以改善,下面我提一下:

1、Item的布局依赖很多Bean的属性,在Node中使用泛型存储与之对应的Bean,这样在getConvertView中就可以通过Node获取到原本的Bean数据了;

2、关于自定义或者不要三角图标;可以让TreeListViewAdapter公布出设置图标的方法,Node全部使用TreeListViewAdapter中设置的图标;关于不显示,直接getConverView里面不管就行了;

3、我们通过注解得到的Id ,pId , label ; 如果嫌慢,可以通过回调的方式进行获取;我们遍历的时候,去通过Adapter中定义类似:abstract int getId(T t) ;将t作为参数,让用户返回id ,类似还有 pid ,label ;这样循环的代码需要从ViewHelper提取到Adapter构造方法中;

4、关于设置包含复选框,选择了多个Node,不要保存position完事,去保存Node中的Id即原Bean的主键;然后在getConvertView中对Id进行对比,防止错乱;

5、关于注解,目前注解只启到了标识的左右;其实还能干很多事,比如默认我们任务用户的id , pid是整形,但是有可能是别的类型;我们可以通过在注解中设置方法来确定,例如:

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TreeNodeId
{
   Class type() ;
}

@TreeNodeId(type = Integer.class)
   private int _id;

当然了,如果你的需求没有上述修改的需要,就不需要折腾了~~

到此,我们整个博客就结束了~~设计中如果存在不足,大家可以自己去改善;希望大家通过本博客学习到的不仅是一个例子如何实现,更多的是如何设计;当然鄙人能力有限,请大家自行去其糟粕;

相关文章
|
1月前
|
算法 Java
算法系列之数据结构-Huffman树
Huffman树(哈夫曼树)又称最优二叉树,是一种带权路径长度最短的二叉树,常用于信息传输、数据压缩等方面。它的构造基于字符出现的频率,通过将频率较低的字符组合在一起,最终形成一棵树。在Huffman树中,每个叶节点代表一个字符,而每个字符的编码则是从根节点到叶节点的路径所对应的二进制序列。
66 3
 算法系列之数据结构-Huffman树
|
1月前
|
存储 自然语言处理 数据库
【数据结构进阶】AVL树深度剖析 + 实现(附源码)
在深入探讨了AVL树的原理和实现后,我们不难发现,这种数据结构不仅优雅地解决了传统二叉搜索树可能面临的性能退化问题,还通过其独特的平衡机制,确保了在任何情况下都能提供稳定且高效的查找、插入和删除操作。
107 19
|
5月前
|
算法
数据结构之博弈树搜索(深度优先搜索)
本文介绍了使用深度优先搜索(DFS)算法在二叉树中执行遍历及构建链表的过程。首先定义了二叉树节点`TreeNode`和链表节点`ListNode`的结构体。通过递归函数`dfs`实现了二叉树的深度优先遍历,按预序(根、左、右)输出节点值。接着,通过`buildLinkedList`函数根据DFS遍历的顺序构建了一个单链表,展示了如何将树结构转换为线性结构。最后,讨论了此算法的优点,如实现简单和内存效率高,同时也指出了潜在的内存管理问题,并分析了算法的时间复杂度。
132 0
|
3月前
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
117 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
3月前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
98 12
|
3月前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
98 10
|
3月前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
126 2
|
5月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
132 5
|
5月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
244 16
|
6月前
|
程序员 开发工具 Android开发
Android|WebView 禁止长按,限制非白名单域名的跳转层级
如何限制 WebView 仅域名白名单网址能随意跳转,并禁用长按选择文字。
105 2
下一篇
oss创建bucket