android如何改变listview的每行高度

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了android如何改变listview的每行高度相关的知识,希望对你有一定的参考价值。

不知道题主的listview单项布局用的是自定义的还是系统的,如果是系统的那就改变不了,系统自带的布局就是类似simple_item这样的,在使用适配器adapter的时候也是new SimpleAdapter 这样。
如果自定义一个listview , 那在单项布局里就可以直接设置它的高度,layout_height="70dp" 。自定义listview的使用需要自己写一个list_item单项布局和相对应的adapter 。
参考技术A listview内的内容格式样式是可以自定义的,在自定义的过程中就可以改变本回答被提问者和网友采纳 参考技术B 不明白你的意思

每行具有不同布局的Android ListView

【中文标题】每行具有不同布局的Android ListView【英文标题】:Android ListView with different layouts for each row 【发布时间】:2011-06-14 05:14:55 【问题描述】:

我正在尝试确定拥有单个 ListView 的最佳方式,该 ListView 包含每行的不同布局。我知道如何创建自定义行 + 自定义数组适配器来支持整个列表视图的自定义行,但是如何在 ListView 中实现许多不同的行样式?

【问题讨论】:

更新:使用 android 的 RecyclerView 进行多行布局的演示 code2concept.blogspot.in/2015/10/… 【参考方案1】:

既然您知道您将拥有多少种布局类型 - 可以使用这些方法。

getViewTypeCount() - 此方法返回您的列表中有多少行类型的信息

getItemViewType(int position) - 根据位置返回您应该使用哪种布局类型的信息

然后,只有当它为 null 并使用 getItemViewType 确定类型时,您才膨胀布局。

查看this tutorial了解更多信息。

为了实现您在评论中描述的结构优化,我建议:

在名为ViewHolder 的对象中存储视图。它将提高速度,因为您不必每次都在 getView 方法中调用 findViewById()。见List14 in API demos。 创建一个通用布局,该布局将符合所有属性组合并在当前位置没有元素时隐藏一些元素。

希望对你有所帮助。如果您可以提供一些带有数据结构的 XML 存根以及您希望如何将其映射到行中的信息,我将能够为您提供更准确的建议。按像素。

【讨论】:

感谢博客非常好,但我添加了复选框。我遇到了一个问题,它将检查第一项并滚动列表。奇怪的匿名项目被检查。你能提供解决方案吗。谢谢 很抱歉再次挖掘这个问题,但您实际上建议使用单个大型布局文件并控制其部分的可见性,而不是使用单独的布局文件,使用 getItemViewType 分别膨胀?跨度> 你也可以这样做。虽然我还是更喜欢这里暴露的方式。它使您想要实现的目标更加清晰。 但是在多布局策略中,我们不能正确地使用视图持有者,因为 setTag 只能包含一个视图持有者,并且每当行布局再次切换时,我们都需要调用 findViewById() 。这使得列表视图的性能非常低。我亲身经历过,您对此有何建议? @pyus13 您可以在单个视图持有者中声明任意数量的视图,并且不必使用视图持有者中声明的每个视图。如果需要示例代码,请告诉我,我会发布。【参考方案2】:

我知道如何创建自定义行 + 自定义数组适配器来支持整个列表视图的自定义行。但是一个列表视图如何支持多种不同的行样式呢?

您已经了解了基础知识。您只需要让您的自定义适配器根据提供的行/光标信息返回不同的布局/视图。

ListView 可以支持多种行样式,因为它派生自 AdapterView:

AdapterView 是一个视图其子级由 Adapter 确定。

如果您查看Adapter,您会看到使用特定行视图的方法:

abstract int getViewTypeCount()
// Returns the number of types of Views that will be created ...

abstract int getItemViewType(int position)
// Get the type of View that will be created ...

abstract View getView(int position, View convertView, ViewGroup parent)
// Get a View that displays the data ...

后两种方法提供位置,因此您可以使用它来确定您的视图类型应该使用该行


当然,您通常不直接使用 AdapterView 和 Adapter,而是使用或派生自它们的子类之一。 Adapter 的子类可能会添加额外的功能来改变如何为不同的行获取自定义布局。 由于用于给定行的视图是由适配器驱动的,因此诀窍是让适配器返回给定行所需的视图。如何做到这一点有所不同取决于具体的适配器。

例如,要使用 ArrayAdapter

覆盖getView() 以膨胀、填充并返回给定位置所需的视图。 getView() 方法包括通过 convertView 参数重用视图的机会。

但要使用 CursorAdapter 的派生词,

覆盖 newView() 以膨胀、填充和返回当前光标状态(即当前“行”)的所需视图 [您还需要覆盖 bindView 以便小部件可以重复使用视图]

但是,要使用 SimpleCursorAdapter

使用setViewValue() 方法定义SimpleCursorAdapter.ViewBinder 以膨胀、填充和返回给定行(当前游标状态)和数据“列”的所需视图。该方法可以只定义“特殊”视图并遵循 SimpleCursorAdapter 的“普通”绑定标准行为。

查找您最终使用的适配器类型的具体示例/教程。

【讨论】:

对于这些适配器类型中的哪一种最适合灵活地实现适配器有什么想法吗?我正在板上为此添加另一个问题。 @Androider - “最适合灵活”是非常开放的 - 没有最终的,可以满足所有需求的类;这是一个丰富的层次结构 - 它归结为子类中是否存在对您的目的有用的功能。如果是这样,从那个子类开始;如果没有,请移至BaseAdapter。从 BaseAdapter 派生将是最“灵活”的,但在代码重用和成熟度方面最差,因为它没有利用已经投入其他适配器的知识和成熟度。 BaseAdapter 用于其他适配器不适合的非标准上下文。 +1 用于区分 CursorAdapterSimpleCursorAdapter 还要注意,如果你重写ArrayAdapter,你给构造函数的布局并不重要,只要getView() 膨胀并返回正确的布局类型 需要注意的是getViewTypeCount()只会在每次调用ListView.setAdapter()时触发一次,而不是每次Adapter.notifyDataSetChanged()时触发一次。【参考方案3】:

看看下面的代码。

首先,我们创建自定义布局。在这种情况下,有四种类型。

even.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_
    android:background="#ff500000"
    android:layout_>

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/white"
        android:layout_
        android:layout_gravity="center"
        android:textSize="24sp"
        android:layout_ />

 </LinearLayout>

奇数.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_
    android:background="#ff001f50"
    android:gravity="right"
    android:layout_>

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/white"
        android:layout_
        android:layout_gravity="center"
        android:textSize="28sp"
        android:layout_  />

 </LinearLayout>

white.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_
    android:background="#ffffffff"
    android:gravity="right"
    android:layout_>

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/black"
        android:layout_
        android:layout_gravity="center"
        android:textSize="28sp"
        android:layout_   />

 </LinearLayout>

black.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_
    android:background="#ff000000"
    android:layout_>

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/white"
        android:layout_
        android:layout_gravity="center"
        android:textSize="33sp"
        android:layout_   />

 </LinearLayout>

然后,我们创建列表视图项。在我们的例子中,带有一个字符串和一个类型。

public class ListViewItem 
        private String text;
        private int type;

        public ListViewItem(String text, int type) 
            this.text = text;
            this.type = type;
        

        public String getText() 
            return text;
        

        public void setText(String text) 
            this.text = text;
        

        public int getType() 
            return type;
        

        public void setType(int type) 
            this.type = type;
        

    

之后,我们创建一个视图持有者。强烈建议这样做,因为 Android 操作系统会保留布局参考,以便在项目消失并重新出现在屏幕上时重用它。如果您不使用这种方法,那么每次您的项目出现在屏幕上时,Android 操作系统都会创建一个新项目并导致您的应用程序泄漏内存。

public class ViewHolder 
        TextView text;

        public ViewHolder(TextView text) 
            this.text = text;
        

        public TextView getText() 
            return text;
        

        public void setText(TextView text) 
            this.text = text;
        

    

最后,我们创建了覆盖 getViewTypeCount() 和 getItemViewType(int position) 的自定义适配器。

public class CustomAdapter extends ArrayAdapter 

        public static final int TYPE_ODD = 0;
        public static final int TYPE_EVEN = 1;
        public static final int TYPE_WHITE = 2;
        public static final int TYPE_BLACK = 3;

        private ListViewItem[] objects;

        @Override
        public int getViewTypeCount() 
            return 4;
        

        @Override
        public int getItemViewType(int position) 
            return objects[position].getType();
        

        public CustomAdapter(Context context, int resource, ListViewItem[] objects) 
            super(context, resource, objects);
            this.objects = objects;
        

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

            ViewHolder viewHolder = null;
            ListViewItem listViewItem = objects[position];
            int listViewItemType = getItemViewType(position);


            if (convertView == null) 

                if (listViewItemType == TYPE_EVEN) 
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_even, null);
                 else if (listViewItemType == TYPE_ODD) 
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_odd, null);
                 else if (listViewItemType == TYPE_WHITE) 
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_white, null);
                 else 
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_black, null);
                

                TextView textView = (TextView) convertView.findViewById(R.id.text);
                viewHolder = new ViewHolder(textView);

                convertView.setTag(viewHolder);

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

            viewHolder.getText().setText(listViewItem.getText());

            return convertView;
        

    

我们的活动是这样的:

private ListView listView;

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main); // here, you can create a single layout with a listview

        listView = (ListView) findViewById(R.id.listview);

        final ListViewItem[] items = new ListViewItem[40];

        for (int i = 0; i < items.length; i++) 
            if (i == 4) 
                items[i] = new ListViewItem("White " + i, CustomAdapter.TYPE_WHITE);
             else if (i == 9) 
                items[i] = new ListViewItem("Black " + i, CustomAdapter.TYPE_BLACK);
             else if (i % 2 == 0) 
                items[i] = new ListViewItem("EVEN " + i, CustomAdapter.TYPE_EVEN);
             else 
                items[i] = new ListViewItem("ODD " + i, CustomAdapter.TYPE_ODD);
            
        

        CustomAdapter customAdapter = new CustomAdapter(this, R.id.text, items);
        listView.setAdapter(customAdapter);
        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() 
            @Override
            public void onItemClick(AdapterView adapterView, View view, int i, long l) 
                Toast.makeText(getBaseContext(), items[i].getText(), Toast.LENGTH_SHORT).show();
            
        );

    

现在在 mainactivity.xml 中创建一个列表视图 像这样

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout 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_
    android:layout_
    android:fitsSystemWindows="true"
    tools:context="com.example.shivnandan.gygy.MainActivity">

    <android.support.design.widget.AppBarLayout
        android:layout_
        android:layout_
        android:theme="@style/AppTheme.AppBarOverlay">

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_
            android:layout_
            android:background="?attr/colorPrimary"
            app:popupTheme="@style/AppTheme.PopupOverlay" />

    </android.support.design.widget.AppBarLayout>

    <include layout="@layout/content_main" />

    <ListView
        android:layout_

        android:layout_

        android:id="@+id/listView"
        android:layout_alignParentRight="true"
        android:layout_alignParentEnd="true"


        android:layout_marginTop="100dp" />

</android.support.design.widget.CoordinatorLayout>

【讨论】:

是从哪里来的 我只需要 (convertView == null) 子句。不需要“viewholder”【参考方案4】:

在您的自定义数组适配器中,您覆盖了 getView() 方法,这您可能很熟悉。然后您所要做的就是使用 switch 语句或 if 语句根据传递给 getView 方法的位置参数返回某个自定义视图。 Android 很聪明,它只会为您的位置/行提供适当类型的 convertView;您无需检查它的类型是否正确。您可以通过适当地覆盖 getItemViewType() 和 getViewTypeCount() 方法来帮助 Android。

【讨论】:

【参考方案5】:

如果我们需要在列表视图中显示不同类型的视图,那么最好在适配器中使用 getViewTypeCount() 和 getItemViewType() 而不是切换视图 VIEW.GONE 和 VIEW.VISIBLE 在 getView( ) 这将影响列表滚动。

Adapter中getViewTypeCount()和getItemViewType()的使用请查看此项。

链接:the-use-of-getviewtypecount

【讨论】:

【参考方案6】:

ListView 旨在用于简单的用例,例如所有行项目的相同静态视图。 由于您必须创建 ViewHolders 并大量使用 getItemViewType(),并动态显示不同的行项目布局 xml,因此您应该尝试使用 Android API 22 中提供的 RecyclerView。它提供了更好的支持和结构多种视图类型。

查看此tutorial,了解如何使用 RecyclerView 来完成您的任务。

【讨论】:

以上是关于android如何改变listview的每行高度的主要内容,如果未能解决你的问题,请参考以下文章

Android ListView固定高度项

每行具有不同布局的Android ListView

Android:如何测量 ListView 的总高度 [重复]

listview使用总结

在 Android 中,如何设置 ListView 项的高度和宽度?

如何在android中以编程方式设置listview高度