Android(安卓)打造任意层级树形控件 考验你的数据结构和设计
1、概述
大家在项目中或多或少的可能会见到,偶尔有的项目需要在APP上显示个树形控件,比如展示一个机构组织,最上面是boss,然后各种部门,各种小boss,最后各种小罗罗;整体是一个树形结构;遇到这样的情况,大家可能回去百度,因为层次多嘛,可能更容易想到ExpandableListView , 因为这玩意层级比Listview多,但是ExpandableListView实现目前只支持两级,当然也有人改造成多级的;但是从我个人角度去看,首先我不喜欢ExpandableListView ,数据集的组织比较复杂。所以今天带大家使用ListView来打造一个树形展示效果。ListView应该是大家再熟悉不过的控件了,并且数据集也就是个List<T> 。
本篇博客目标实现,只要是符合树形结构的数据可以轻松的通过我们的代码,实现树形效果,有多轻松,文末就知道了~~
好了,既然是要展现树形结构,那么数据上肯定就是树形的一个依赖,也就是说,你的每条记录,至少有个字段指向它的父节点;类似(id , pId, others ....)
2、原理分析
先看看我们的效果图:
我们支持任意层级,包括item的布局依然让用户自己的去控制,我们的demo的Item布局很简单,一个图标+文本~~
原理就是,树形不树形,其实不就是多个缩进么,只要能够判断每个item属于树的第几层(术语貌似叫高度),设置合适的缩进即可。
当然了,原理说起来简单,还得控制每一层间关系,添加展开缩回等,以及有了缩进还要能显示在正确的位置,不过没关系,我会带着大家一步一步实现的。
3、用法
由于整体比较长,我决定首先带大家看一下用法,就是如果学完了这篇博客,我们需要树形控件,我们需要花多少精力去完成~~
现在需求来了:我现在需要展示一个文件管理系统的树形结构:
数据是这样的:
[html] view plain copy
- //id,pid,label,其他属性
- mDatas.add(newFileBean(1,0,"文件管理系统"));
- mDatas.add(newFileBean(2,1,"游戏"));
- mDatas.add(newFileBean(3,1,"文档"));
- mDatas.add(newFileBean(4,1,"程序"));
- mDatas.add(newFileBean(5,2,"war3"));
- mDatas.add(newFileBean(6,2,"刀塔传奇"));
- mDatas.add(newFileBean(7,4,"面向对象"));
- mDatas.add(newFileBean(8,4,"非面向对象"));
- mDatas.add(newFileBean(9,7,"C++"));
- mDatas.add(newFileBean(10,7,"JAVA"));
- mDatas.add(newFileBean(11,7,"Javascript"));
- mDatas.add(newFileBean(12,8,"C"));
当然了,bean可以有很多属性,我们提供你动态的设置树节点上的显示、以及不约束id, pid 的命名,你可以起任意丧心病狂的属性名称;
那么我们如何确定呢?
看下Bean:
[java] view plain copy
- packagecom.zhy.bean;
- importcom.zhy.tree.bean.TreeNodeId;
- importcom.zhy.tree.bean.TreeNodeLabel;
- importcom.zhy.tree.bean.TreeNodePid;
- publicclassFileBean
- {
- @TreeNodeId
- privateint_id;
- @TreeNodePid
- privateintparentId;
- @TreeNodeLabel
- privateStringname;
- privatelonglength;
- privateStringdesc;
- publicFileBean(int_id,intparentId,Stringname)
- {
- super();
- this._id=_id;
- this.parentId=parentId;
- this.name=name;
- }
- }
现在,不用说,应该也知道我们通过注解来确定的。
下面看我们如何将这数据转化为树
布局文件就一个listview,就补贴了,直接看Activity
[java] view plain copy
- packagecom.zhy.tree_view;
- importjava.util.ArrayList;
- importjava.util.List;
- importandroid.app.Activity;
- importandroid.os.Bundle;
- importandroid.widget.ListView;
- importcom.zhy.bean.FileBean;
- importcom.zhy.tree.bean.TreeListViewAdapter;
- publicclassMainActivityextendsActivity
- {
- privateList<FileBean>mDatas=newArrayList<FileBean>();
- privateListViewmTree;
- privateTreeListViewAdaptermAdapter;
- @Override
- protectedvoidonCreate(BundlesavedInstanceState)
- {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- initDatas();
- mTree=(ListView)findViewById(R.id.id_tree);
- try
- {
- mAdapter=newSimpleTreeAdapter<FileBean>(mTree,this,mDatas,10);
- mTree.setAdapter(mAdapter);
- }catch(IllegalAccessExceptione)
- {
- e.printStackTrace();
- }
- }
- privatevoidinitDatas()
- {
- //id,pid,label,其他属性
- mDatas.add(newFileBean(1,0,"文件管理系统"));
- mDatas.add(newFileBean(2,1,"游戏"));
- mDatas.add(newFileBean(3,1,"文档"));
- mDatas.add(newFileBean(4,1,"程序"));
- mDatas.add(newFileBean(5,2,"war3"));
- mDatas.add(newFileBean(6,2,"刀塔传奇"));
- mDatas.add(newFileBean(7,4,"面向对象"));
- mDatas.add(newFileBean(8,4,"非面向对象"));
- mDatas.add(newFileBean(9,7,"C++"));
- mDatas.add(newFileBean(10,7,"JAVA"));
- mDatas.add(newFileBean(11,7,"Javascript"));
- mDatas.add(newFileBean(12,8,"C"));
- }
- }
Activity里面并没有什么特殊的代码,拿到listview,传入mData,当中初始化了一个Adapter;
看来我们的核心代码都在我们的Adapter里面:
那么看一眼我们的Adapter
[java] view plain copy
- packagecom.zhy.tree_view;
- importjava.util.List;
- importandroid.content.Context;
- importandroid.view.View;
- importandroid.view.ViewGroup;
- importandroid.widget.ImageView;
- importandroid.widget.ListView;
- importandroid.widget.TextView;
- importcom.zhy.tree.bean.Node;
- importcom.zhy.tree.bean.TreeListViewAdapter;
- publicclassSimpleTreeAdapter<T>extendsTreeListViewAdapter<T>
- {
- publicSimpleTreeAdapter(ListViewmTree,Contextcontext,List<T>datas,
- intdefaultExpandLevel)throwsIllegalArgumentException,
- IllegalAccessException
- {
- super(mTree,context,datas,defaultExpandLevel);
- }
- @Override
- publicViewgetConvertView(Nodenode,intposition,ViewconvertView,ViewGroupparent)
- {
- ViewHolderviewHolder=null;
- if(convertView==null)
- {
- convertView=mInflater.inflate(R.layout.list_item,parent,false);
- viewHolder=newViewHolder();
- 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());
- returnconvertView;
- }
- privatefinalclassViewHolder
- {
- ImageViewicon;
- TextViewlabel;
- }
- }
我们的SimpleTreeAdapter继承了我们的TreeListViewAdapter ; 除此之外,代码上只需要复写getConvertView , 且getConvetView其实和我们平时的getView写法一致;
公布出getConvertView 的目的是,让用户自己去决定Item的展示效果。其他的代码,我已经打包成jar了,用的时候导入即可。这样就完成了我们的树形控件。
也就是说用我们的树形控件,只需要将传统继承BaseAdapter改为我们的TreeListViewAdapter ,然后去实现getConvertView 就好了。
那么现在的效果是:
默认就全打开了,因为我们也支持动态设置打开的层级,方面使用者使用。
用起来是不是很随意,加几个注解,ListView的Adapater换个类继承下~~好了,下面开始带大家一起从无到有的实现~
4、实现
1、思路
我们的思路是这样的,我们显示时,需要很多属性,我们需要知道当前节点是否是父节点,当前的层级,他的孩子节点等等;但是用户的数据集是不固定的,最多只能给出类似id,pId 这样的属性。也就是说,用户给的bean并不适合我们用于控制显示,于是我们准备这样做:
1、在用户的Bean中提取出必要的几个元素 id , pId , 以及显示的文本(通过注解+反射);然后组装成我们的真正显示时的Node;即List<Bean> -> List<Node>
2、显示的并非是全部的Node,比如某些节点的父节点是关闭状态,我们需要进行过滤;即List<Node> ->过滤后的List<Node>
3、显示时,比如点击父节点,它的子节点会跟随其后显示,我们内部是个List,也就是说,这个List的顺序也是很关键的;当然排序我们可以放为步骤一;
最后将过滤后的Node进行显示,设置左内边距即可。
说了这么多,首先看一眼我们封装后的Node
2、Node
[java] view plain copy
- packagecom.zhy.tree.bean;
- importjava.util.ArrayList;
- importjava.util.List;
- importorg.w3c.dom.NamedNodeMap;
- importandroid.util.Log;
- publicclassNode
- {
- privateintid;
- /**
- *根节点pId为0
- */
- privateintpId=0;
- privateStringname;
- /**
- *当前的级别
- */
- privateintlevel;
- /**
- *是否展开
- */
- privatebooleanisExpand=false;
- privateinticon;
- /**
- *下一级的子Node
- */
- privateList<Node>children=newArrayList<Node>();
- /**
- *父Node
- */
- privateNodeparent;
- publicNode()
- {
- }
- publicNode(intid,intpId,Stringname)
- {
- super();
- this.id=id;
- this.pId=pId;
- this.name=name;
- }
- publicintgetIcon()
- {
- returnicon;
- }
- publicvoidsetIcon(inticon)
- {
- this.icon=icon;
- }
- publicintgetId()
- {
- returnid;
- }
- publicvoidsetId(intid)
- {
- this.id=id;
- }
- publicintgetpId()
- {
- returnpId;
- }
- publicvoidsetpId(intpId)
- {
- this.pId=pId;
- }
- publicStringgetName()
- {
- returnname;
- }
- publicvoidsetName(Stringname)
- {
- this.name=name;
- }
- publicvoidsetLevel(intlevel)
- {
- this.level=level;
- }
- publicbooleanisExpand()
- {
- returnisExpand;
- }
- publicList<Node>getChildren()
- {
- returnchildren;
- }
- publicvoidsetChildren(List<Node>children)
- {
- this.children=children;
- }
- publicNodegetParent()
- {
- returnparent;
- }
- publicvoidsetParent(Nodeparent)
- {
- this.parent=parent;
- }
- /**
- *是否为跟节点
- *
- *@return
- */
- publicbooleanisRoot()
- {
- returnparent==null;
- }
- /**
- *判断父节点是否展开
- *
- *@return
- */
- publicbooleanisParentExpand()
- {
- if(parent==null)
- returnfalse;
- returnparent.isExpand();
- }
- /**
- *是否是叶子界点
- *
- *@return
- */
- publicbooleanisLeaf()
- {
- returnchildren.size()==0;
- }
- /**
- *获取level
- */
- publicintgetLevel()
- {
- returnparent==null?0:parent.getLevel()+1;
- }
- /**
- *设置展开
- *
- *@paramisExpand
- */
- publicvoidsetExpand(booleanisExpand)
- {
- this.isExpand=isExpand;
- if(!isExpand)
- {
- for(Nodenode:children)
- {
- node.setExpand(isExpand);
- }
- }
- }
- }
包含了树节点一些常见的属性,一些常见的方法;对于getLevel,setExpand这些方法,大家可以好好看看~
有了Node,刚才的用法中,出现的就是我们Adapter所继承的超类:TreeListViewAdapter;核心代码都在里面,我们准备去一探究竟:
3、TreeListViewAdapter
代码不是很长,直接完整的贴出:
[java] view plain copy
- packagecom.zhy.tree.bean;
- importjava.util.List;
- importandroid.content.Context;
- importandroid.view.LayoutInflater;
- importandroid.view.View;
- importandroid.view.ViewGroup;
- importandroid.widget.AdapterView;
- importandroid.widget.AdapterView.OnItemClickListener;
- importandroid.widget.BaseAdapter;
- importandroid.widget.ListView;
- publicabstractclassTreeListViewAdapter<T>extendsBaseAdapter
- {
- protectedContextmContext;
- /**
- *存储所有可见的Node
- */
- protectedList<Node>mNodes;
- protectedLayoutInflatermInflater;
- /**
- *存储所有的Node
- */
- protectedList<Node>mAllNodes;
- /**
- *点击的回调接口
- */
- privateOnTreeNodeClickListeneronTreeNodeClickListener;
- publicinterfaceOnTreeNodeClickListener
- {
- voidonClick(Nodenode,intposition);
- }
- publicvoidsetOnTreeNodeClickListener(
- OnTreeNodeClickListeneronTreeNodeClickListener)
- {
- this.onTreeNodeClickListener=onTreeNodeClickListener;
- }
- /**
- *
- *@parammTree
- *@paramcontext
- *@paramdatas
- *@paramdefaultExpandLevel
- *默认展开几级树
- *@throwsIllegalArgumentException
- *@throwsIllegalAccessException
- */
- publicTreeListViewAdapter(ListViewmTree,Contextcontext,List<T>datas,
- intdefaultExpandLevel)throwsIllegalArgumentException,
- IllegalAccessException
- {
- mContext=context;
- /**
- *对所有的Node进行排序
- */
- mAllNodes=TreeHelper.getSortedNodes(datas,defaultExpandLevel);
- /**
- *过滤出可见的Node
- */
- mNodes=TreeHelper.filterVisibleNode(mAllNodes);
- mInflater=LayoutInflater.from(context);
- /**
- *设置节点点击时,可以展开以及关闭;并且将ItemClick事件继续往外公布
- */
- mTree.setOnItemClickListener(newOnItemClickListener()
- {
- @Override
- publicvoidonItemClick(AdapterView<?>parent,Viewview,
- intposition,longid)
- {
- expandOrCollapse(position);
- if(onTreeNodeClickListener!=null)
- {
- onTreeNodeClickListener.onClick(mNodes.get(position),
- position);
- }
- }
- });
- }
- /**
- *相应ListView的点击事件展开或关闭某节点
- *
- *@paramposition
- */
- publicvoidexpandOrCollapse(intposition)
- {
- Noden=mNodes.get(position);
- if(n!=null)//排除传入参数错误异常
- {
- if(!n.isLeaf())
- {
- n.setExpand(!n.isExpand());
- mNodes=TreeHelper.filterVisibleNode(mAllNodes);
- notifyDataSetChanged();//刷新视图
- }
- }
- }
- @Override
- publicintgetCount()
- {
- returnmNodes.size();
- }
- @Override
- publicObjectgetItem(intposition)
- {
- returnmNodes.get(position);
- }
- @Override
- publiclonggetItemId(intposition)
- {
- returnposition;
- }
- @Override
- publicViewgetView(intposition,ViewconvertView,ViewGroupparent)
- {
- Nodenode=mNodes.get(position);
- convertView=getConvertView(node,position,convertView,parent);
- //设置内边距
- convertView.setPadding(node.getLevel()*30,3,3,3);
- returnconvertView;
- }
- publicabstractViewgetConvertView(Nodenode,intposition,
- ViewconvertView,ViewGroupparent);
- }
首先我们的类继承自BaseAdapter,然后我们对应的数据集是,过滤出的可见的Node;
我们的构造方法默认接收4个参数:listview,context,mdatas,以及默认展开的级数:0只显示根节点;
可以在构造方法中看到:对用户传入的数据集做了排序,和过滤的操作;一会再看这些方法,这些方法我们使用了一个TreeHelper进行了封装。
注:如果你觉得你的Item布局十分复杂,且布局会展示Bean的其他数据,那么为了方便,你可以让Node中包含一个泛型T , 每个Node携带与之对于的Bean的所有数据;
可以看到我们还直接为Item设置了点击事件,因为我们树,默认就有点击父节点展开与关闭;但是为了让用户依然可用点击监听,我们自定义了一个点击的回调供用户使用;
当用户点击时,默认调用expandOrCollapse方法,将当然节点重置展开标志,然后重新过滤出可见的Node,最后notifyDataSetChanged即可;
其他的方法都是BaseAdapter默认的一些方法了。
下面我们看下TreeHelper中的一些方法:
4、TreeHelper
首先看TreeListViewAdapter构造方法中用到的两个方法:
[java] view plain copy
- /**
- *传入我们的普通bean,转化为我们排序后的Node
- *@paramdatas
- *@paramdefaultExpandLevel
- *@return
- *@throwsIllegalArgumentException
- *@throwsIllegalAccessException
- */
- publicstatic<T>List<Node>getSortedNodes(List<T>datas,
- intdefaultExpandLevel)throwsIllegalArgumentException,
- IllegalAccessException
- {
- List<Node>result=newArrayList<Node>();
- //将用户数据转化为List<Node>以及设置Node间关系
- List<Node>nodes=convetData2Node(datas);
- //拿到根节点
- List<Node>rootNodes=getRootNodes(nodes);
- //排序
- for(Nodenode:rootNodes)
- {
- addNode(result,node,defaultExpandLevel,1);
- }
- returnresult;
- }
拿到用户传入的数据,转化为List<Node>以及设置Node间关系,然后根节点,从根往下遍历进行排序;
接下来看:filterVisibleNode
[java] view plain copy
- /**
- *过滤出所有可见的Node
- *
- *@paramnodes
- *@return
- */
- publicstaticList<Node>filterVisibleNode(List<Node>nodes)
- {
- List<Node>result=newArrayList<Node>();
- for(Nodenode:nodes)
- {
- //如果为跟节点,或者上层目录为展开状态
- if(node.isRoot()||node.isParentExpand())
- {
- setNodeIcon(node);
- result.add(node);
- }
- }
- returnresult;
- }
过滤Node的代码很简单,遍历所有的Node,只要是根节点或者父节点是展开状态就添加返回;
最后看看这两个方法用到的别的一些私有方法:
[java] view plain copy
- /**
- *将我们的数据转化为树的节点
- *
- *@paramdatas
- *@return
- *@throwsNoSuchFieldException
- *@throwsIllegalAccessException
- *@throwsIllegalArgumentException
- */
- privatestatic<T>List<Node>convetData2Node(List<T>datas)
- throwsIllegalArgumentException,IllegalAccessException
- {
- List<Node>nodes=newArrayList<Node>();
- Nodenode=null;
- for(Tt:datas)
- {
- intid=-1;
- intpId=-1;
- Stringlabel=null;
- Class<?extendsObject>clazz=t.getClass();
- Field[]declaredFields=clazz.getDeclaredFields();
- for(Fieldf: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=newNode(id,pId,label);
- nodes.add(node);
- }
- /**
- *设置Node间,父子关系;让每两个节点都比较一次,即可设置其中的关系
- */
- for(inti=0;i<nodes.size();i++)
- {
- Noden=nodes.get(i);
- for(intj=i+1;j<nodes.size();j++)
- {
- Nodem=nodes.get(j);
- if(m.getpId()==n.getId())
- {
- n.getChildren().add(m);
- m.setParent(n);
- }elseif(m.getId()==n.getpId())
- {
- m.getChildren().add(n);
- n.setParent(m);
- }
- }
- }
- //设置图片
- for(Noden:nodes)
- {
- setNodeIcon(n);
- }
- returnnodes;
- }
- privatestaticList<Node>getRootNodes(List<Node>nodes)
- {
- List<Node>root=newArrayList<Node>();
- for(Nodenode:nodes)
- {
- if(node.isRoot())
- root.add(node);
- }
- returnroot;
- }
- /**
- *把一个节点上的所有的内容都挂上去
- */
- privatestaticvoidaddNode(List<Node>nodes,Nodenode,
- intdefaultExpandLeval,intcurrentLevel)
- {
- nodes.add(node);
- if(defaultExpandLeval>=currentLevel)
- {
- node.setExpand(true);
- }
- if(node.isLeaf())
- return;
- for(inti=0;i<node.getChildren().size();i++)
- {
- addNode(nodes,node.getChildren().get(i),defaultExpandLeval,
- currentLevel+1);
- }
- }
- /**
- *设置节点的图标
- *
- *@paramnode
- */
- privatestaticvoidsetNodeIcon(Nodenode)
- {
- if(node.getChildren().size()>0&&node.isExpand())
- {
- node.setIcon(R.drawable.tree_ex);
- }elseif(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
[java] view plain copy
- packagecom.zhy.tree.bean;
- importjava.lang.annotation.ElementType;
- importjava.lang.annotation.Retention;
- importjava.lang.annotation.RetentionPolicy;
- importjava.lang.annotation.Target;
- @Target(ElementType.FIELD)
- @Retention(RetentionPolicy.RUNTIME)
- public@interfaceTreeNodeId
- {
- }
TreeNodePid
[java] view plain copy
- packagecom.zhy.tree.bean;
- importjava.lang.annotation.ElementType;
- importjava.lang.annotation.Retention;
- importjava.lang.annotation.RetentionPolicy;
- importjava.lang.annotation.Target;
- @Target(ElementType.FIELD)
- @Retention(RetentionPolicy.RUNTIME)
- public@interfaceTreeNodePid
- {
- }
[java] view plain copy
- packagecom.zhy.tree.bean;
- importjava.lang.annotation.ElementType;
- importjava.lang.annotation.Retention;
- importjava.lang.annotation.RetentionPolicy;
- importjava.lang.annotation.Target;
- @Target(ElementType.FIELD)
- @Retention(RetentionPolicy.RUNTIME)
- public@interfaceTreeNodeLabel
- {
- }
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是整形,但是有可能是别的类型;我们可以通过在注解中设置方法来确定,例如:
[java] view plain copy
- @Target(ElementType.FIELD)
- @Retention(RetentionPolicy.RUNTIME)
- public@interfaceTreeNodeId
- {
- Classtype();
- }
[java] view plain copy
- @TreeNodeId(type=Integer.class)
- privateint_id;
当然了,如果你的需求没有上述修改的需要,就不需要折腾了~~
到此,我们整个博客就结束了~~设计中如果存在不足,大家可以自己去改善;希望大家通过本博客学习到的不仅是一个例子如何实现,更多的是如何设计;当然鄙人能力有限,请大家自行去其糟粕;
源码上传不了,上传了半天都有问题!大家需要的话可以评论或者私信联系我。
更多相关文章
- Android搜索控件的基本使用方法
- ART深度探索开篇:从Method Hook谈起
- Android(安卓)模仿QQ抢红包 listView实现
- 为什么我的Android(安卓)Studio没有Android(安卓)SDK选项
- Android与WebView的同步和异步访问机制
- Activity生命周期详解
- Android中的HashMap原理实践探索,重写equals(),为什么重写hashCode
- 浅谈Java中Collections.sort对List排序的两种方法
- Python list sort方法的具体使用