具有任意数量行类型的 ListView 适配器(不知道不同行类型的数量)

Posted

技术标签:

【中文标题】具有任意数量行类型的 ListView 适配器(不知道不同行类型的数量)【英文标题】:ListView Adapter with arbitrary number of row types (Don't know the number of different row types) 【发布时间】:2013-06-26 13:41:25 【问题描述】:

所以,我正在制作这个应用程序。该应用程序解析一个网站,或者更具体地说是一个 vbulletin-board。当我在论坛中解析一个主题时,我已经将其划分,以便当我解析该主题中的每个帖子时,我会在诸如此类的部分中获得帖子的实际内容,并以正确的顺序存储这些部分在数组中:

[Plain text]
[Quote from somebody]
[Plain text]
[Another quote]
[Another quote again]
[Some more plain text]

但是,帖子可以按照您可能知道的任何顺序排列,并且可以包含比示例中更多或更少的部分,并且它也不必在其中包含引号,或者它可能只是一个或几个引号。一切皆有可能。

当我在我的应用程序中列出帖子时,我使用的是 ListView。然后,此列表视图的每一行将始终由一个标题和前面提到的部分的任意组合组成。

在谷歌上搜索了一下之后我想这样做的方式是在一个 XML 文件中有一个只有一个布局标签的“基本布局”,并且每个部分都有一个单独的布局,存储在单独的 XML 中-files,并在我的适配器中每次调用 getView() 时,查看我的“帖子列表”中该位置的帖子,然后循环浏览该特定帖子中的部分,并膨胀一个新的“报价布局”对于存储在帖子中的每个引用部分,并为帖子中的每个纯文本部分膨胀一个“纯文本布局”。对于每个人,我都会填写属于该帖子的所有内容。

我认为这可行,但可能存在性能问题?据我了解,布局膨胀非常昂贵,而且我也无法回收传入 getView() 的 View,因为它可能添加了一堆部分,而我在另一个 getView 调用中可能不需要这些部分().. 也就是说,如果我对 getView() 和回收有所了解。

这是适配器的 getView() 方法的基本示例:

@Override
public View getView(int i, View view, ViewGroup viewGroup) 
    // Inflate the base-layout, which the others are added to.
    view = mInflater.inflate(R.layout.forum_post,null);

    View header = mInflater.inflate(R.layout.post_header_layout, null);
    View message = mInflater.inflate(R.layout.post_text_layout, null);
    View quote = mInflater.inflate(R.layout.post_quote_layout, null);

    ((ViewGroup)view).addView(header);
    ((ViewGroup)view).addView(message);
    ((ViewGroup)view).addView(quote);
    return view;

然后当我从已保存的帖子列表中提取数据时,根据需要增加更多的引用视图/消息视图。

基本布局只是一个线性布局标签

我膨胀的布局只是添加了一些 TextView 和 ImageView 的 RelativeLayouts。

这段代码产生了这个结果,其中我有一个 Header 用户名、图片等,一段纯文本,一段引用段。 不过,这似乎并不能一直正常工作,因为当我刚刚尝试时,列表的副本似乎卡在了背景上,而另一个在其上滚动了..

http://s14.postimg.org/rizid8q69/view.png

有没有更好的方法来做到这一点?因为我认为这不是很有效

【问题讨论】:

您想在列表视图中为您的行添加不同的布局吗? 【参考方案1】:

您需要覆盖getViewItemTypegetViewTypeCount

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

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

例子:

   private static final int TYPE_ITEM1 = 0;
   private static final int TYPE_ITEM2 = 1;
   private static final int TYPE_ITEM3 = 2;
    @Override; 
    public int getItemViewType(int position) 
    
    int type;
    if (position== 0) // your condition
        type = TYPE_ITEM1;  //type 0 for header
     else if(position == 1)
        type = TYPE_ITEM2; //type 1 for message
    else 
        type = TYPE_ITEM3; //type 2 for Quote
    
    return type;
    
@Override
public int getViewTypeCount() 
    return 3;    //three different layouts to be inflated

getView

 int type= getItemViewType(i); // determine type using position.
 switch (type) 
 case TYPE_ITEM1: 
    view= mInflater.inflate(R.layout.post_header_layout, null);   // inflate layout for header
 break;
 case TYPE_ITEM2:
     view = mInflater.inflate(R.layout.post_text_layout, null); //   inflate layout for quote
 break;
 case TYPE_ITEM3:
      quote = mInflater.inflate(R.layout.post_quote_layout, null);  //   inflate layout for message
 break;    
 .... 

您需要使用 View Holder 来实现平滑滚动和性能。

http://developer.android.com/training/improving-layouts/smooth-scrolling.html

您可以查看下面的教程

http://android.amberfog.com/?p=296

【讨论】:

@user1944184 却不知道如何扩展布局? 我想你误会了?因为当我知道我有两种不同类型的行时,我在另一个适配器中使用了这两种方法。但是这次 getItemViewType() 中的每个类型都不是 ListView 中的单独行,它们是 ListView 中一行中的单独行,并且 ListView 行中可以有任意数量的 type2 和 type3 行.已经检查了教程,但它没有帮助,因为我列表中的行不是一组不同类型的行也许我没有很好地解释我的问题.. 也许我必须以编程方式代替 @user1944184 哇!。我真的不明白你的问题。抱歉帮不上忙。 我之前的回复是点击错误,你还记得你的回答吗?编辑:好的:D【参考方案2】:

首先,您要重用已作为参数之一传递的convertView。这样你就可以避免给 View 充气。

其次,您可以使用ViewHolder 来存储对您内部Views 的引用。使用ViewHolder 将提高性能,无论您是膨胀视图还是通过 id 查找它们,因为这两种方法都非常昂贵。

ViewHolder 设置为项目View 上的标签。

@Override
public View getView(int position, View convertView, ViewGroup parent) 
    View view;
    ViewHolder viewHolder;

    // if possible reuse view
    if (convertView == null) 
        final LayoutInflater inflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        view = inflater.inflate(resource, parent, false);
        viewHolder = new ViewHolder(mInflater.inflate(R.layout.post_header_layout, null));
        view.setTag(viewHolder);
     else 
        // reuse view
        view = convertView;
        viewHolder = (ViewHolder) view.getTag();
    

    //set text, listeners, icon, etc.

    return view;

ViewHolder 只是私有内部类存储引用到视图。

私有静态类 ViewHolder

    private final View view;

    private ViewHolder(View view) 
        this.view = view;
    

ListView 的用法在Google IO 2010 上给出。

【讨论】:

【参考方案3】:

inflater 需要知道未来父 ViewGroup 的真实类型,因此下面的代码是错误的:

view = mInflater.inflate(R.layout.forum_post,null);

相反,你应该使用这个:

view = mInflater.inflate(R.layout.forum_post,viewGroup,false);

对于另一个inflate,同样的事情:使用真正的父级(在这种情况下为view)或另一个与(未来)父级具有相同类型的viewGroup;否则 LayoutParameters 将不会设置为正确的类型,并且您在 XML 代码中指定的值将丢失(从不使用)。

【讨论】:

以上是关于具有任意数量行类型的 ListView 适配器(不知道不同行类型的数量)的主要内容,如果未能解决你的问题,请参考以下文章

每行具有不同布局的Android ListView

具有多种布局的Android ListView

Python 3 类型,具有任意数量的包含类型的自定义可变参数泛型,如何?

在listview中的addTextChangedListener中使用自定义列表适配器

具有包含 CheckBoxes 的自定义适配器的 Listview

Listview onItemClickListener 问题适配器与适配器