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

时间:2013-06-28 17:35:02

标签: android android-listview android-adapter

所以,我正在制作这个应用程序。该应用程序解析一个网站,或更具体地说是一个vbulletin-board。当我在论坛中解析一个帖子时,我把它分开了,这样当我解析该帖子中的每个帖子时,我会在这样的部分中得到帖子的实际内容,并且我按照正确的顺序存储这些部分在数组中:

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

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

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

在谷歌搜索之后,我想要做的就是在一个XML文件中只有一个布局标签的“基础布局”,每个部分都有一个单独的布局,存储在单独的XML中-files,并在每次调用我的适配器中的getView()时,查看我的“Post-list”中该位置的帖子,然后遍历该特定帖子中的各个部分,并膨胀一个新的“Quote-layout”对于存储在帖子中的每个引用部分,并为帖子中的每个纯文本部分膨胀“纯文本布局”。对于每个人,我填写属于该帖子的所有内容。

我认为这样可行,但可能存在性能问题?据我所知,布局膨胀是非常昂贵的,我也无法将传入的视图回收到getView(),因为它可能会添加一些部分,我可能不需要再调用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;
}

当我从保存的帖子列表中提取数据时,根据需要扩展更多的quote-views / message-views。

基本布局只是一个LinearLayout-tag

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

此代码生成此结果,其中包含标题 用户名,图片等。,纯文本的一部分,以及一个引用部分。 这似乎并不是一直都能正常工作,因为当我刚才尝试时,列表的副本似乎卡在了背景上而另一个在它上面滚动了..

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

有更好的方法吗?因为我认为这不是很有效率

3 个答案:

答案 0 :(得分:5)

您需要覆盖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

答案 1 :(得分:0)

首先,您要重用已作为参数之一传递的convertView。这样您就可以避免让项目View膨胀。

其次,您可以使用ViewHolder之类的内容来存储对内部View的引用。使用ViewHolder可以提高性能,无论您是通过id膨胀视图还是通过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;
    }
}

Google IO 2010处讨论ListView使用情况。

答案 2 :(得分:0)

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

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

相反,你应该使用这个:

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

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