所以,我正在制作这个应用程序。该应用程序解析一个网站,或更具体地说是一个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
有更好的方法吗?因为我认为这不是很有效率
答案 0 :(得分:5)
您需要覆盖getViewItemType
和getViewTypeCount
。
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
您可以查看以下教程
答案 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代码中指定的值将丢失(从未使用过)。