情景: 有两个分类,每个分类下的数据用listview显示,分类1需要显示头视图和尾视图,分类2不要显示头视图和尾视图 。两个分类可以来回切换。



实现的步骤: 1. addHeaderView 、 addFooterView,默认显示分类1的数据,setAdapter绑定 

2. 点击分类2时,先removeheaderview、removefooterview,然后setAdapter重新绑定数据

3. 点击分类1时,addHeaderView、addFooterView,然后setAdapter重新绑定数据

修改之前的代码:



问题解决:listview多次调用addHeaderView和addFooterView_查看源码

public void addHeaderFooter(){ 
if(!hasHead){
listview.addHeaderView(tv_head);
hasHead = true;
}
if(!hasFoot){
listview.addFooterView(tv_foot);
hasHead = false;
}
}
@Override
public void onClick(int position) {
Log.d(TAG, "position " + position + " is clicked!");
switch(position){
case 1:
//不需要header和footer
if(hasHead){
listview.removeHeaderView(tv_head);
hasHead = false;
}
if(hasFoot){
listview.removeFooterView(tv_foot);
hasFoot = false;
}
break;
default:
//其余分类都需要
addHeaderFooter();
break;
}


问题解决:listview多次调用addHeaderView和addFooterView_查看源码


遇到的问题: 

步骤3中addHeaderView时报错:

问题解决:listview多次调用addHeaderView和addFooterView_查看源码_03

 

解决办法:

add之前, setAdapter(null).

查看源码可知:

问题解决:listview多次调用addHeaderView和addFooterView_ide_04

 

将adapter置null后,就不会报错了


这样改之后又遇到了一个问题:

步骤3中addFooterView时报错:

问题解决:listview多次调用addHeaderView和addFooterView_数据_05

追踪源码可知:

问题解决:listview多次调用addHeaderView和addFooterView_查看源码_06

此处getAdapter为null(之前调用了setAdapter(null)),所以nullpointer了

为什么会走到这一步呢,查看addFooterView的源码:

问题解决:listview多次调用addHeaderView和addFooterView_数据_07

调用了mDataSetObserver.onChanged(),

而这个 mDataSetObserver 在哪里呢?见setAdapter源码:

问题解决:listview多次调用addHeaderView和addFooterView_数据_08

原来,在步骤1中调用了setAdapter设置数据,所以这里的mDataSetObserver就存在了。

所以在setAdapter(null)之后,调用addFooterView时,由于mDataSetObserver依然存在,就会调用mDataSetObserver.onChanged()方法,从而调用getAdapter方法,返回null,再调用getcount时就nullpointer了。

解决办法: 我是在addFooterView之前又绑定了一次,setAdapter,给了一个空的数据源进去,只是确保adapter不为null。 这个方法有点投巧,没找到好的方法。


改正后的代码:



问题解决:listview多次调用addHeaderView和addFooterView_查看源码

public void addHeaderFooter(){ 
listview.setAdapter(null);
if(!hasHead){
listview.addHeaderView(tv_head);
hasHead = true;
}
if(!hasFoot){
SimpleAdapter adapter = new SimpleAdapter(this, empty_list, R.layout.funtest_listview_item,
new String[]{ FuntestColumns.name,FuntestColumns.r_count,FuntestColumns.description},
new int[]{R.id.funtest_name, R.id.funtest_hot, R.id.funtest_description});
listview.setAdapter(adapter);
listview.addFooterView(tv_foot);
hasHead = false;
}
}
@Override
public void onClick(int position) {
Log.d(TAG, "position " + position + " is clicked!");
switch(position){
case 1:
//不需要header和footer
if(hasHead){
listview.removeHeaderView(tv_head);
hasHead = false;
}
if(hasFoot){
listview.removeFooterView(tv_foot);
hasFoot = false;
}
break;
default:
//其余分类都需要
addHeaderFooter();
break;
}


问题解决:listview多次调用addHeaderView和addFooterView_查看源码


 

总结: 多次调用addHeaderView、addFooterView时

1. addHeaderView之前调用setAdapter(null)

2. addFooterView之前调用setAdapter(adapter); //adapter中传入空数据源

实现的步骤: 1. addHeaderView 、 addFooterView,默认显示分类1的数据,setAdapter绑定 

2. 点击分类2时,先removeheaderview、removefooterview,然后setAdapter重新绑定数据

3. 点击分类1时,addHeaderView、addFooterView,然后setAdapter重新绑定数据

修改之前的代码:



问题解决:listview多次调用addHeaderView和addFooterView_查看源码

public void addHeaderFooter(){ 
if(!hasHead){
listview.addHeaderView(tv_head);
hasHead = true;
}
if(!hasFoot){
listview.addFooterView(tv_foot);
hasHead = false;
}
}
@Override
public void onClick(int position) {
Log.d(TAG, "position " + position + " is clicked!");
switch(position){
case 1:
//不需要header和footer
if(hasHead){
listview.removeHeaderView(tv_head);
hasHead = false;
}
if(hasFoot){
listview.removeFooterView(tv_foot);
hasFoot = false;
}
break;
default:
//其余分类都需要
addHeaderFooter();
break;
}


问题解决:listview多次调用addHeaderView和addFooterView_查看源码


遇到的问题: 

步骤3中addHeaderView时报错:

问题解决:listview多次调用addHeaderView和addFooterView_查看源码_03

 

解决办法:

add之前, setAdapter(null).

查看源码可知:

问题解决:listview多次调用addHeaderView和addFooterView_ide_04

 

将adapter置null后,就不会报错了


这样改之后又遇到了一个问题:

步骤3中addFooterView时报错:

问题解决:listview多次调用addHeaderView和addFooterView_数据_05

追踪源码可知:

问题解决:listview多次调用addHeaderView和addFooterView_查看源码_06

此处getAdapter为null(之前调用了setAdapter(null)),所以nullpointer了

为什么会走到这一步呢,查看addFooterView的源码:

问题解决:listview多次调用addHeaderView和addFooterView_数据_07

调用了mDataSetObserver.onChanged(),

而这个 mDataSetObserver 在哪里呢?见setAdapter源码:

问题解决:listview多次调用addHeaderView和addFooterView_数据_08

原来,在步骤1中调用了setAdapter设置数据,所以这里的mDataSetObserver就存在了。

所以在setAdapter(null)之后,调用addFooterView时,由于mDataSetObserver依然存在,就会调用mDataSetObserver.onChanged()方法,从而调用getAdapter方法,返回null,再调用getcount时就nullpointer了。

解决办法: 我是在addFooterView之前又绑定了一次,setAdapter,给了一个空的数据源进去,只是确保adapter不为null。 这个方法有点投巧,没找到好的方法。


改正后的代码:



问题解决:listview多次调用addHeaderView和addFooterView_查看源码

public void addHeaderFooter(){ 
listview.setAdapter(null);
if(!hasHead){
listview.addHeaderView(tv_head);
hasHead = true;
}
if(!hasFoot){
SimpleAdapter adapter = new SimpleAdapter(this, empty_list, R.layout.funtest_listview_item,
new String[]{ FuntestColumns.name,FuntestColumns.r_count,FuntestColumns.description},
new int[]{R.id.funtest_name, R.id.funtest_hot, R.id.funtest_description});
listview.setAdapter(adapter);
listview.addFooterView(tv_foot);
hasHead = false;
}
}
@Override
public void onClick(int position) {
Log.d(TAG, "position " + position + " is clicked!");
switch(position){
case 1:
//不需要header和footer
if(hasHead){
listview.removeHeaderView(tv_head);
hasHead = false;
}
if(hasFoot){
listview.removeFooterView(tv_foot);
hasFoot = false;
}
break;
default:
//其余分类都需要
addHeaderFooter();
break;
}


问题解决:listview多次调用addHeaderView和addFooterView_查看源码


 

总结: 多次调用addHeaderView、addFooterView时

1. addHeaderView之前调用setAdapter(null)

2. addFooterView之前调用setAdapter(adapter); //adapter中传入空数据源