从零开始水安卓——高级UI组件2(ListView下)

 

BaseAdapter

使用BaseAdapter来实现更灵活的列表,由于是一个抽象类,需要写一个适配器继承该类。

依然是ListView上部分提到的四个方法。

布局

布局和前面基本没有差别,需要注意的是也需要单独定义一个布局文件listitem.xml(即你的自定义布局,名字随意啦)

listitem.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center_vertical">
 
    <ImageView
        android:id="@+id/imageView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginLeft="10dp"
        android:layout_weight="1"
        tools:srcCompat="@tools:sample/avatars[0]" />
 
    <TextView
        android:id="@+id/textView"
        android:gravity="center_vertical"
        android:layout_width="wrap_content"
        android:layout_height="138dp"
        android:layout_weight="1"
        android:text="TextView" />
 
</LinearLayout>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">


    <ListView
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:layout_marginStart="8dp"
        android:layout_marginLeft="8dp"
        android:layout_marginTop="8dp"
        android:layout_marginEnd="8dp"
        android:layout_marginRight="8dp"
        android:layout_marginBottom="8dp"
        android:id="@+id/listview"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>

MainActivity

同样需要一个字符串数组,int类型数组(图片) ?当然还得准备相应的图片资源,放在drawable文件夹即可

四个方法的参数和之前基本一致,所以自定义的内容也基本类似,需要注意的同样需要一个构造方法,来传递上下文

private Context context;
public MyAdapter(Context context){
    this.context = context;
}

同样的

getView()

方法需要获取Textview和ImageView并将定义好的两个数组填充进去,最后返回view

完整代码:

package com.example.listview2;

import android.content.Context;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listview);
        listView.setAdapter(new MyAdapter(this));
    }

    //自定义适配器
    static class MyAdapter extends BaseAdapter{
        private String[] titles ={
                "title-1",
                "title-2",
                "title-3",
                "title-4",
                "title-5",
        };
        private int[]icons={
                R.drawable.a1,
                R.drawable.a2,
                R.drawable.a3,
                R.drawable.a4,
                R.drawable.a5,
        };
        private Context context;
        public MyAdapter(Context context){
            this.context = context;
        }
        @Override
        public int getCount() {
            return titles.length;
        }

        @Override
        public Object getItem(int position) {
            return titles[position];
        }

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

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            LayoutInflater inflater = LayoutInflater.from(context);
            View view =inflater.inflate(R.layout.listitem,null);
            TextView tv = view.findViewById(R.id.textView);
            ImageView iv = view.findViewById(R.id.imageView);

            tv.setText(titles[position]);
            iv.setImageResource(icons[position]);

            return view;
        }
    }
}

效果

 

ListView性能优化

1、重复使用convertView

以?的为例

在MainActivity的中的

getView()中

添加一条System.out.println来进行测试

可以看到屏幕上有三个列表项,所以打印了三条语句。

如果滑动屏幕...则会反复打印当前屏幕上的列表项对应的view...

即不断的创建view对象...可以看到0就出现了两次,后面的地址都是不一样的。

综上,这样就造成了很大的浪费,为了解决这个问题。

首先把ListView布局中的宽高改成match_parent

(但是其实再这个项目里好像没啥用...)

?关键

利用convertView进行缓存,当convertView为空的时候,才去实例化view。

(convertView也是view所以也不用再单独定义一个view了)

可以看到 为1的条目地址是一样的,即没有重新创建对象,这样就不会有浪费了。

优化后完整代码:

package com.example.listview2;

import android.content.Context;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listview);
        listView.setAdapter(new MyAdapter(this));
    }

    //自定义适配器
    static class MyAdapter extends BaseAdapter{
        private String[] titles ={
                "title-1",
                "title-2",
                "title-3",
                "title-4",
                "title-5",
        };
        private int[]icons={
                R.drawable.a1,
                R.drawable.a2,
                R.drawable.a3,
                R.drawable.a4,
                R.drawable.a5,
        };
        private Context context;
        public MyAdapter(Context context){
            this.context = context;
        }
        @Override
        public int getCount() {
            return titles.length;
        }

        @Override
        public Object getItem(int position) {
            return titles[position];
        }

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

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {

            if(convertView == null){
                LayoutInflater inflater = LayoutInflater.from(context);
                //实例化一个布局文件
                convertView =inflater.inflate(R.layout.listitem,null);
            }

            TextView tv = convertView.findViewById(R.id.textView);
            ImageView iv = convertView.findViewById(R.id.imageView);
            System.out.println(position+"----   "+convertView);
            tv.setText(titles[position]);
            iv.setImageResource(icons[position]);

            return convertView;
        }
    }
}

 

2、使用ViewHolder提高在容器中查找组件的效率

?的方法虽然可以解决重新创建对象的问题,但是依旧有重复调用的现象,即上下滚动界面,还是会出现若干个1的条目。

 

关键在于将findViewById的组件和view绑定

 

新建一个内部类ViewHolder

声明你的组件 ,用于保存第一次查找的组件

取代原来的直接定义TextView、ImageView

并且利用

convertView.setTag(vh);

进行存储

convertView.getTag();

进行读取

 

最终修改版——完整代码:

package com.example.listview2;

import android.content.Context;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listview);
        listView.setAdapter(new MyAdapter(this));
    }

    //自定义适配器
    static class MyAdapter extends BaseAdapter{
        private String[] titles ={
                "title-1",
                "title-2",
                "title-3",
                "title-4",
                "title-5",
        };
        private int[]icons={
                R.drawable.a1,
                R.drawable.a2,
                R.drawable.a3,
                R.drawable.a4,
                R.drawable.a5,
        };
        private Context context;
        public MyAdapter(Context context){
            this.context = context;
        }
        @Override
        public int getCount() {
            return titles.length;
        }

        @Override
        public Object getItem(int position) {
            return titles[position];
        }

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

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewHolder vh;
            if(convertView == null){
                LayoutInflater inflater = LayoutInflater.from(context);
                //实例化一个布局文件
                convertView =inflater.inflate(R.layout.listitem,null);
                vh = new ViewHolder();
                vh.tv = convertView.findViewById(R.id.textView);
                vh.iv = convertView.findViewById(R.id.imageView);
                convertView.setTag(vh);
            }else
            {
                vh = (ViewHolder) convertView.getTag();
            }

            vh.tv.setText(titles[position]);
            vh.iv.setImageResource(icons[position]);
            return convertView;
        }
        //用于保存第一次查找的组件,避免下次重复查找
        static class ViewHolder{
            ImageView iv;
            TextView tv;
        }
    }
}

ListView刷新分页

 

准备

当ListView的选项非常多的时候,就需要分页了。下面先准备实验基本的代码,流程的话和?基本一致

需要一个子布局listitem.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="title"
        android:layout_marginLeft="10dp"
        android:id="@+id/tv_title"
        android:textSize="50sp"
        />
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="content"
        android:layout_marginLeft="10dp"
        android:textSize="50sp"
        android:id="@+id/tv_content"
        />
</LinearLayout>

需要一个News类

package com.example.listview_fenye;

public class News {
    String title;
    String content;
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <ListView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/listview"
        >
    </ListView>

</android.support.constraint.ConstraintLayout>

MainActivity 

package com.example.listview_fenye;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;

import java.util.Vector;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    //容器装载对象
    private Vector<News> news = new Vector<>();
    private MyAdapter myAdapter;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listview);
        initData();
        myAdapter = new MyAdapter();
        listView.setAdapter(myAdapter);
    }
    private int index = 1;
    //初始化数据
    private void initData(){
        //每次初始化10条
        for(int i=0;i<10;i++){
            News n  = new News();
            n.title = "title:"+index;
            n.content = "content:"+index;
            index++;
            news.add(n);
        }
    }
    //创建一个Adapater,方便直接用 news 起见没加static 实际上应该加上
    class MyAdapter extends BaseAdapter{

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

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

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

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewHolder vh;
            if(convertView==null){
                convertView = getLayoutInflater().inflate(R.layout.listitem,null);
                vh = new ViewHolder();
                vh.tv_content = convertView.findViewById(R.id.tv_content);
                vh.tv_title = convertView.findViewById(R.id.tv_title);
                convertView.setTag(vh);
            }else{
                vh = (ViewHolder) convertView.getTag();
            }
            News n = news.get(position);
            vh.tv_title.setText(n.title);
            vh.tv_content.setText(n.content);
            return convertView;
        }
        class ViewHolder{
            TextView tv_title;
            TextView tv_content;
        }
    }
}

 

基本效果:

 

Loading

下面先实现加载数据的提示,需要单独新建一个布局

通过listView.addFooterView来加到界面中去(或者...Head...)

 

所以要先准备一个loading布局(loading.xml,名字无所谓啦)

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <ProgressBar
        android:id="@+id/progressBar"
        style="?android:attr/progressBarStyle"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_weight="1" />

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="51dp"
        android:layout_weight="1"
        android:text="正在加载中..." 
        android:gravity="center_vertical"/>

</LinearLayout>

效果的话..可能有点丑,哎呀算了..将就一下

MainActivity修改后的代码的话 在下面的部分一起放好了...不然要贴两次

然后用线程模拟加载

线程

写一个线程类(继承Thred)

class LoadDataThread extends Thread{
    @Override
    public void run(){
        initData();//再加10条数据
        try {
            Thread.sleep(2000);//休眠2s
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

滚动条事件

 implements AbsListView.OnScrollListener

重写其两个方法

onScrollStateChanged(AbsListView view, int scrollState)  状态改变...

部分参数参数解析

scrollState:具有如下三个状态:
分别是正在滚动(手已经离开了)、空闲的状态、手还在滚动条上的状态

onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) 滚动状态发生变化..

部分参数参数解析:

firstVisibleItem 第一个可显示的选项

visibleItemCount 可显示的选项总量

totalItemCount 选项总量

当前两个参数等于最后一个参数时,即滚动到最后

 

判断滚动到最底下进行加载。

 

这里有一个注意事项

即:

不能写成?

所以需要用到

Handler机制
用到其handleMessage(Message msg)方法来处理消息

再在子线程中使用

sendEmptyMessage()来发送

完整代码:

package com.example.listview_fenye;

import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AbsListView;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;

import java.util.Vector;

public class MainActivity extends AppCompatActivity implements AbsListView.OnScrollListener {
    private ListView listView;
    //容器装载对象
    private Vector<News> news = new Vector<>();
    private MyAdapter myAdapter;
    //数据更新完成后的标记 0x1即16进制的1
    private static final int DAYA_UPDATE = 0x1;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listview);
        listView.setOnScrollListener(this);
        View footview = getLayoutInflater().inflate(R.layout.loading,null);
        listView.addFooterView(footview);
        initData();
        myAdapter = new MyAdapter();
        listView.setAdapter(myAdapter);
    }
    private int index = 1;
    //初始化数据
    private void initData(){
        //每次初始化10条
        for(int i=0;i<10;i++){
            News n  = new News();
            n.title = "title:"+index;
            n.content = "content:"+index;
            index++;
            news.add(n);
        }
    }
    private int visibleLastIndex;//用来可显示的最后一条数据的索引
    @Override
    public void onScrollStateChanged(AbsListView view, int scrollState) {
        if(myAdapter.getCount()==visibleLastIndex && scrollState == AbsListView.OnScrollListener.SCROLL_STATE_IDLE){
            new  LoadDataThread().start(); //启动线程
        }
    }

    @Override
    public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
        visibleLastIndex = firstVisibleItem+visibleItemCount-1; //减1是因为还有个正在加载中...
    }
    //线程间通讯机制
    private Handler handler = new Handler(){
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what){
                case DAYA_UPDATE:
                    myAdapter.notifyDataSetChanged();//由主线程自己更新消息
                    break;
            }
        }
    };
    //线程类
    class LoadDataThread extends Thread{
        @Override
        public void run(){
            initData();//再加10条数据
            try {
                Thread.sleep(2000);//休眠2s
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //子线程不能直接访问UI组件
           // myAdapter.notifyDataSetChanged();
            //通过handler给主线程发送一个消息标记
            handler.sendEmptyMessage(DAYA_UPDATE);
        }
    }
    //创建一个Adapater,方便直接用 news 起见没加static 实际上应该加上
    class MyAdapter extends BaseAdapter{

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

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

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

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewHolder vh;
            if(convertView==null){
                convertView = getLayoutInflater().inflate(R.layout.listitem,null);
                vh = new ViewHolder();
                vh.tv_content = convertView.findViewById(R.id.tv_content);
                vh.tv_title = convertView.findViewById(R.id.tv_title);
                convertView.setTag(vh);
            }else{
                vh = (ViewHolder) convertView.getTag();
            }
            News n = news.get(position);
            vh.tv_title.setText(n.title);
            vh.tv_content.setText(n.content);
            return convertView;
        }
        class ViewHolder{
            TextView tv_title;
            TextView tv_content;
        }
    }
}

效果如图:

即可以无限制向下加载新的数据

 

ExpendableListView

概述

expandable刚好翻译为可扩展的,在视图中显示垂直滚动两级列表。和ListView还是有一定区别的,允许且仅允许两个层次:组织单独可以扩展为其子项。

(总而言之,就是能实现二级菜单)

实现流程

1、设置数据源

2、设置适配器,ExpandableAdapter类为ExpandableList的一个子类,需要写一个继承BaseExpandableListAdapter,

方法很多,所以需要重写的方法也很多,如下:

//hasStableIds()在本例中不用

虽然要重写的方法很多,但是根据方法名,基本上也能理解方法的含义,看下面部分的代码直接就可以理解了。

关键在于下面两个方法的重写

@Override
public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) {
    return null;
}

@Override
public View getChildView(int groupPosition, int childPosition, boolean isLastChild, View convertView, ViewGroup parent) {
    return null;
}

为了完成两个方法,所以需要建立一个布局group_layout.xml和child_layout.xml,两个布局基本上一样就好(但最好还是有点区别,不要完全一样,不然有点奇怪)

 

具体代码

MainActivity

package com.example.expandblelistview;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseExpandableListAdapter;
import android.widget.ExpandableListAdapter;
import android.widget.ExpandableListView;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
    private ExpandableListView listView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.expandablelistview);
        listView.setAdapter(new MyExpandableAdapter());
        //点击事件
        listView.setOnChildClickListener(new ExpandableListView.OnChildClickListener() {
            @Override
            public boolean onChildClick(ExpandableListView parent, View v, int groupPosition, int childPosition, long id) {
                Toast.makeText(MainActivity.this,childs[groupPosition][childPosition],Toast.LENGTH_LONG).show();
                return true;
            }
        });
    }
    //提供一组数据
    private String[]groups={"分组1","分组2"};
    private String[][] childs={{"测试1","测试2","测试3"},{"测试9","测试8","测试7"}};

    //适配器
    class MyExpandableAdapter extends BaseExpandableListAdapter {

        @Override
        public int getGroupCount() {
            return groups.length;
        }

        @Override
        public int getChildrenCount(int groupPosition) {
            return childs[groupPosition].length;
        }

        @Override
        public Object getGroup(int groupPosition) {
            return groups[groupPosition];
        }

        @Override
        public Object getChild(int groupPosition, int childPosition) {
            return childs[groupPosition][childPosition];
        }

        @Override
        public long getGroupId(int groupPosition) {
            return groupPosition;
        }

        @Override
        public long getChildId(int groupPosition, int childPosition) {
            return childPosition;
        }

        @Override
        public boolean hasStableIds() {
            return false;
        }

        @Override
        public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) {
           if(convertView==null){
               convertView = getLayoutInflater().inflate(R.layout.group_layout,null);
           }
           ImageView icon = convertView.findViewById(R.id.icon);
            TextView title = convertView.findViewById(R.id.title);

            title.setText(groups[groupPosition]);
            return convertView;
        }

        @Override
        public View getChildView(int groupPosition, int childPosition, boolean isLastChild, View convertView, ViewGroup parent) {
            if(convertView==null){
                convertView = getLayoutInflater().inflate(R.layout.child_layout,null);
            }
            ImageView icon = convertView.findViewById(R.id.icon);
            TextView title = convertView.findViewById(R.id.title);

            title.setText(childs[groupPosition][childPosition]);
            return convertView;
        }

        @Override
        public boolean isChildSelectable(int groupPosition, int childPosition) {
            return true;
        }
    }
}

group_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center_vertical">
    <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:maxWidth="64dp"
        android:maxHeight="64dp"
        android:id="@+id/icon"
        android:adjustViewBounds="true"
        android:src="@mipmap/ic_launcher"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/title"
        android:text="not data"/>
</LinearLayout>

child_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:paddingLeft="30dp"
    android:gravity="center_vertical">
    <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:maxWidth="64dp"
        android:maxHeight="64dp"
        android:id="@+id/icon"
        android:adjustViewBounds="true"
        android:src="@mipmap/ic_launcher"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/title"
        android:text="not data"/>
</LinearLayout>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <ExpandableListView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/expandablelistview"
        >
    </ExpandableListView>
</android.support.constraint.ConstraintLayout>

 

效果

点击分组后?点击子菜单?

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Go语言(也称为Golang)是由Google开发的一种静态强类型、编译型的编程语言。它旨在成为一门简单、高效、安全和并发的编程语言,特别适用于构建高性能的服务器和分布式系统。以下是Go语言的一些主要特点和优势: 简洁性:Go语言的语法简单直观,易于学习和使用。它避免了复杂的语法特性,如继承、重载等,转而采用组合和接口来实现代码的复用和扩展。 高性能:Go语言具有出色的性能,可以媲美C和C++。它使用静态类型系统和编译型语言的优势,能够生成高效的机器码。 并发性:Go语言内置了对并发的支持,通过轻量级的goroutine和channel机制,可以轻松实现并发编程。这使得Go语言在构建高性能的服务器和分布式系统时具有天然的优势。 安全性:Go语言具有强大的类型系统和内存管理机制,能够减少运行时错误和内存泄漏等问题。它还支持编译时检查,可以在编译阶段就发现潜在的问题。 标准库:Go语言的标准库非常丰富,包含了大量的实用功能和工具,如网络编程、文件操作、加密解密等。这使得开发者可以更加专注于业务逻辑的实现,而无需花费太多时间在底层功能的实现上。 跨平台:Go语言支持多种操作系统和平台,包括Windows、Linux、macOS等。它使用统一的构建系统(如Go Modules),可以轻松地跨平台编译和运行代码。 开源和社区支持:Go语言是开源的,具有庞大的社区支持和丰富的资源。开发者可以通过社区获取帮助、分享经验和学习资料。 总之,Go语言是一种简单、高效、安全、并发的编程语言,特别适用于构建高性能的服务器和分布式系统。如果你正在寻找一种易于学习和使用的编程语言,并且需要处理大量的并发请求和数据,那么Go语言可能是一个不错的选择。
Go语言(也称为Golang)是由Google开发的一种静态强类型、编译型的编程语言。它旨在成为一门简单、高效、安全和并发的编程语言,特别适用于构建高性能的服务器和分布式系统。以下是Go语言的一些主要特点和优势: 简洁性:Go语言的语法简单直观,易于学习和使用。它避免了复杂的语法特性,如继承、重载等,转而采用组合和接口来实现代码的复用和扩展。 高性能:Go语言具有出色的性能,可以媲美C和C++。它使用静态类型系统和编译型语言的优势,能够生成高效的机器码。 并发性:Go语言内置了对并发的支持,通过轻量级的goroutine和channel机制,可以轻松实现并发编程。这使得Go语言在构建高性能的服务器和分布式系统时具有天然的优势。 安全性:Go语言具有强大的类型系统和内存管理机制,能够减少运行时错误和内存泄漏等问题。它还支持编译时检查,可以在编译阶段就发现潜在的问题。 标准库:Go语言的标准库非常丰富,包含了大量的实用功能和工具,如网络编程、文件操作、加密解密等。这使得开发者可以更加专注于业务逻辑的实现,而无需花费太多时间在底层功能的实现上。 跨平台:Go语言支持多种操作系统和平台,包括Windows、Linux、macOS等。它使用统一的构建系统(如Go Modules),可以轻松地跨平台编译和运行代码。 开源和社区支持:Go语言是开源的,具有庞大的社区支持和丰富的资源。开发者可以通过社区获取帮助、分享经验和学习资料。 总之,Go语言是一种简单、高效、安全、并发的编程语言,特别适用于构建高性能的服务器和分布式系统。如果你正在寻找一种易于学习和使用的编程语言,并且需要处理大量的并发请求和数据,那么Go语言可能是一个不错的选择。
Go语言(也称为Golang)是由Google开发的一种静态强类型、编译型的编程语言。它旨在成为一门简单、高效、安全和并发的编程语言,特别适用于构建高性能的服务器和分布式系统。以下是Go语言的一些主要特点和优势: 简洁性:Go语言的语法简单直观,易于学习和使用。它避免了复杂的语法特性,如继承、重载等,转而采用组合和接口来实现代码的复用和扩展。 高性能:Go语言具有出色的性能,可以媲美C和C++。它使用静态类型系统和编译型语言的优势,能够生成高效的机器码。 并发性:Go语言内置了对并发的支持,通过轻量级的goroutine和channel机制,可以轻松实现并发编程。这使得Go语言在构建高性能的服务器和分布式系统时具有天然的优势。 安全性:Go语言具有强大的类型系统和内存管理机制,能够减少运行时错误和内存泄漏等问题。它还支持编译时检查,可以在编译阶段就发现潜在的问题。 标准库:Go语言的标准库非常丰富,包含了大量的实用功能和工具,如网络编程、文件操作、加密解密等。这使得开发者可以更加专注于业务逻辑的实现,而无需花费太多时间在底层功能的实现上。 跨平台:Go语言支持多种操作系统和平台,包括Windows、Linux、macOS等。它使用统一的构建系统(如Go Modules),可以轻松地跨平台编译和运行代码。 开源和社区支持:Go语言是开源的,具有庞大的社区支持和丰富的资源。开发者可以通过社区获取帮助、分享经验和学习资料。 总之,Go语言是一种简单、高效、安全、并发的编程语言,特别适用于构建高性能的服务器和分布式系统。如果你正在寻找一种易于学习和使用的编程语言,并且需要处理大量的并发请求和数据,那么Go语言可能是一个不错的选择。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

云无心鸟知还

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值