[zzerX@blog ~ ]:

ViewPager添加ListView遇到的findViewById为null

初次使用viewPager就踩了坑,首先我的需求是在一个viewPager中添加3个layout 其中一个layout为Listview

各文件:
activity_main.xml

<RelativeLayout
	xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
   	android:layout_width="match_parent"
        android:layout_height="match_parent">
	<androidx.viewpager.widget.ViewPager
	android:layout_width="match_parent"
	android:layout_height="match_parent"/>
</RelativeLayout>

my_list_view.xml

<LinearLayout
	xmlns:android="http://schemas.android.com/apk/res/android"
	android:layout_width="fill_parent"
	android:layout_height="fill_parent"
	android:orientation="vertical">
	<ListView		
	android:id="@+id/mylistview"
        android:layout_width="wrap_content"
	style="?android:attr/buttonStyleSmall"
	android:layout_height="wrap_content"/>
</LinearLayout>

MyPagerAdapter.java

public class MyPagerAdapter extends PagerAdapter {

    private List<View> mViewList;

    public MyPagerAdapter(List<View> mViewList) {
        this.mViewList = mViewList;
    }

    @Override
    public int getCount() {//必须实现
        return mViewList.size();
    }

    @Override
    public boolean isViewFromObject(View view, Object object) {//必须实现
        return view == object;
    }

    @Override
    public Object instantiateItem(ViewGroup container, int position) {//必须实现,实例化
        container.addView(mViewList.get(position));
        return mViewList.get(position);
    }

    @Override
    public void destroyItem(ViewGroup container, int position, Object object) {//必须实现,销毁
        container.removeView(mViewList.get(position));
    }



}

MainActivity.java
为了省事,这里只在Viewpager中添加了一个layout

public class MainActivity extends AppCompatActivity {
   private View view1; 
   MyPagerAdapter pagerAdapter;    
   private ViewPager mViewPager;
   List<View> viewList;//view数组 
   private ListView list;  
  @Override
    protected void onCreate(Bundle savedInstanceState) {     
      super.onCreate(savedInstanceState);  
      setContentView(R.layout.activity_main);
       mViewPager = findViewById(R.id.activityViewPager); 
       LayoutInflater inflater=getLayoutInflater();   
       view1 = inflater.inflate(R.layout.my_list_view,null);     
       viewList = new ArrayList<View>();// 将要分页显示的View装入数组中
       viewList.add(view1);    
       pagerAdapter = new MyPagerAdapter(viewList);   
       mViewPager.setAdapter(pagerAdapter);          //添加listview
       list = findViewById(R.id.mylistview);   
       String data[] = {"aa","bb","cc","dd","aa","bb","cc","dd","aa","bb","cc","dd","aa","bb","cc","dd","uu"};//假数据 
          //添加并且显示      
       ArrayAdapter<String> adapter = new ArrayAdapter<String(this,android.R.layout.simple_list_item_1,data);//新建并配置ArrayAapeter   
         list.setAdapter(adapter); 
             }
 }

ok,运行,发现报错了

AndroidRuntime
java.lang.NullPointerException: Attempt to invoke virtual method ‘android.view.View android.view.View.findViewById(int)’ on a null object reference

findViewById的是个空的对象,怎么会,检查了my_list_view.xml中的Listview id确实没错,那问题一定出在findViewById上!

查阅了相关资料发现,MainActivity中直接的findViewById其实是this.findViewbyId

代表的是setContentView加载的xml中去find,可是我们的activity_main.xml里面只写了ViewPager,

ListView是动态添加上去的,所以他是空的对象,我要获取my_list_view.xml中的Listview怎么办?

使用ViewPager时我们还用到了另一个方法inflate,
使用这个就能获取到Listview的id!
来试试:

list = getLayoutInflater().inflate(R.layout.my_listview,null).findViewById(R.id.mylistview); 

编译!运行!没报错!咦?但是ViewPager居然是空白的,

明明已经在listview中添加了数据才对啊.
我再一次看了getLayoutInflater().inflate方法
inflate的作用就是将一个用xml定义的布局文件查找出来,是加载一个布局文件,然后查找id,而我们的Listview在前面的ViewPager中已经被加载了呀!

所以,我们需要从前面加载的ListView中去findView,而不是从一个新的inflate方法中取findView,导致了无数据的情况。

所以我们应该改成:

list = view1.findViewById(R.id.mylistview); 

大功告成,原来这么简单。

总结

findViewById有上下文关系,需要理解需要find的上文是谁,才能准确的找到相关view

— Feb 19, 2020