Фрагмент в Activity получает значение context null в setAdapter для ListView

#android #listview #android-fragments #android-asynctask #fragment

#Android #listview #android-фрагменты #android-asynctask #фрагмент

Вопрос:

это проблема, с которой я сталкиваюсь уже около 3 месяцев, на самом деле, с тех пор, как я начал разрабатывать это приложение. Я прочитал сотни сообщений на этом веб-сайте, в других я читал учебные пособия и не нашел решения. Я считаю, что эта проблема связана с тем, как у меня устроена структура приложения.

Я разрабатываю приложение, которое отображает несколько разделов, каждый раздел представляет собой фрагмент, а основное действие использует ViewPager для отображения по одному фрагменту за раз (например, приложение fb или приложение Twitter). В каждом разделе будут отображаться списки с разной информацией: новости, видео, объявления и многое другое. Как только отображается раздел, скажем, раздел новостей, он показывает listview со статьями / новостями. Проблема заключается в том, когда я пытаюсь отобразить listview, потому что в адаптере, который я использую для отображения listview, выдает мне исключение NullPointerException в строке setAdapter .

Я отследил строку, которая выдает мне исключение и фактически является context объявлением. Несмотря на то, что контекст равен нулю во всех моих фрагментах (я не знаю почему), я могу отображать listview, но в некоторых других приложение выходит из строя и выдает NPE. Я получаю контекст через строку Context context=getActivity() , но когда я тестирую это, он дает мне null.

Это один из фрагментов, который я успешно отобразил в listview

 public class HomeActivity extends Fragment{
   static ListView newsList;
   static Item item=null;
   Context context=getActivity();
   ItemAdapter itemAdapter;
   List<Item> news=new ArrayList<Item>();
   static int i=0;
   static Activity activity;
   static String HOME_FRAGMENT="NEWS";
   public static final int TIME_ENTRY_REQUEST_CODE=1;
   FragmentManager fm;


public View onCreateView(LayoutInflater inflater,ViewGroup container,Bundle savedInstanceState){
   super.onCreateView(inflater, container, savedInstanceState);
   View view=inflater.inflate(R.layout.home_activity, container,false);
   newsList=(ListView)view.findViewById(R.id.newslist);
   setRetainInstance(true); 
   return view;
}

public void onAttach(Activity activity){
    super.onAttach(activity);
    HomeActivity.activity=activity;
}

public void onSaveInstanceState(Bundle outState){
    super.onSaveInstanceState(outState);
}

public void displayNews(final List<Item> news){
    try {        
         if(news==null || news.size()==0){
                //Retrieve again the data list
         }
         Log.d("HomeActivity","total news: " news.size());
     } catch (Exception e) {
         e.printStackTrace();
     }
    if(context==null){
         Log.d("HomeActivity","context null!");
    }
    itemAdapter=new ItemAdapter(context,(ArrayList<Item>) news);
    newsList.setAdapter(itemAdapter);
}   
  

}

ОБНОВЛЕНО: и это еще один фрагмент, в котором я получаю NPE,

 public class VideosActivity extends Fragment{

ListView listvideo;
ItemAdapter itemAdapter;
Context context;
Activity activity;
public View onCreateView(LayoutInflater inflater,ViewGroup container,Bundle savedInstanceState){
       super.onCreateView(inflater, container, savedInstanceState);
       View view=inflater.inflate(R.layout.videos_activity, container,false);
       listvideo=(ListView)view.findViewById(R.id.videoslist);
       Log.d("videos activity","onCreateView ready!!");
               context=getActivity();
       setRetainInstance(true); 
       return view;
    }


public void onAttach(Activity activity){
    super.onAttach(activity);
    this.activity=activity;
}

public void displayVideos(final List<Item> videos){
    try {        
         if(videos==null || videos.size()==0){
                //Retrieve again the data list

         }
     } catch (Exception e) {
         e.printStackTrace();
     }
    for(int i=0;i<videos.size();i  ){

    }
    itemAdapter=new ItemAdapter(activity,(ArrayList<Item>) videos);//I also used context in this line, but I still get the NPE
    listvideo.setAdapter(itemAdapter); //Here's the line that gives me the NPE
}   
  

}

I am parsing all the info from an XML, and I’m managing everything through AsyncTask
This is my adapter

 public class ItemAdapter extends BaseAdapter implements OnItemClickListener{
    Item item=new Item();
    private ArrayList<Item> news=new ArrayList<Item>();
    private static final int NO_PICTURE_VIEW=0;
    private static final int PICTURE_VIEW=1;

    public ItemAdapter(Context context,ArrayList<Item> items) {
    super();
    news.addAll(items);
    }
    public int getCount() {
    // TODO Auto-generated method stub
    return news.size();
}

    public Item getItem(int index) {
    // TODO Auto-generated method stub
    return getItem(index);
}

    public long getItemId(int index) {
    // TODO Auto-generated method stub
    return index;
}

    public int getViewTypeCount() {
         return 2;
}

    public int getItemViewType(int position) {
    Item article=news.get(position);      
        if(article.getImage()!=null)      
            return PICTURE_VIEW;
        else
                return NO_PICTURE_VIEW;
}
@Override
    public View getView(int index, View view, ViewGroup parent) {
    item=news.get(index);
    int type=getItemViewType(index);
    if(view==null){
            LayoutInflater inflater=LayoutInflater.from(parent.getContext());
        if(type==NO_PICTURE_VIEW){
        view=inflater.inflate(R.layout.item_no_picture_list,parent,false);
                TextView titleView=(TextView)view.findViewById(R.id.titleNoPicture);
        titleView.setText(item.getTitle());
        }
        else{
        view=inflater.inflate(R.layout.item_picture_list,parent,false);
        TextView titleView=(TextView)view.findViewById(R.id.titleArticle);
        ImageView image=(ImageView)view.findViewById(R.id.imageItem);
        titleView.setText(item.getTitle());
        image.setImageBitmap(item.getImage());
    }
}

return view;
}
}
  

This is the main activity

 public class MainActivity extends FragmentActivity{
    private String tabs[]={"News","Sports","Valley Life","Videos","Classifieds","Directory","Weather"};
ViewPager viewPager=null;
FragmentManager fragmentManager=getSupportFragmentManager();
android.support.v4.app.FragmentTransaction ft=fragmentManager.beginTransaction();
    AdapterView adapterView;
static ParseManager parser=new ParseManager();
static List<Item> news=new ArrayList<Item>();
ArrayList<Item> sports=new ArrayList<Item>();
ArrayList<Item> valleyLife=new ArrayList<Item>();
public static final int TIME_ENTRY_REQUEST_CODE=1;
HomeActivity home;
JSONManager jmanager=new JSONManager();
WeatherManager weather=new WeatherManager();

    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    viewPager=(ViewPager)findViewById(R.id.pager);
    viewPager.setOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener(){
                public void onPageSelected(int position)          
                getActionBar().setSelectedNavigationItem(position);
            }
    });
    viewPager.setAdapter(new AdapterView(fragmentManager));
    viewPager.setOffscreenPageLimit(tabs.length);
    adapterView=new AdapterView(getSupportFragmentManager());
    /*initialization of the action bar: color, icon amp; title.*/
    final android.app.ActionBar actionbar=getActionBar();
    ColorDrawable color=new ColorDrawable(Color.parseColor("#cd853f"));
    actionbar.setBackgroundDrawable(color);
    actionbar.setTitle("Title");
    ActionBar.TabListener tabListener=new ActionBar.TabListener() {
        public void onTabUnselected(Tab tab, FragmentTransaction ft) {

        }   


        @Override
        public void onTabSelected(Tab tab, FragmentTransaction ft) {
            viewPager.setCurrentItem(tab.getPosition());
            ft=getFragmentManager().beginTransaction();
            //ft.replace(layouts[tab.getPosition()], (adapterView.getItem(tab.getPosition()));
            ft.commit();

        }

        @Override
        public void onTabReselected(Tab tab, FragmentTransaction ft) {  

        }
    };
    /*Displaying Tabs for the app */
    for(int i=0;i<tabs.length;i  )  
        actionbar.addTab(actionbar.newTab().setText(tabs[i]).setTabListener(tabListener));
    actionbar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
            viewPager.setOnPageChangeListener(new ViewPager.OnPageChangeListener() {
                @Override
                public void onPageSelected(int position) {
                actionbar.setSelectedNavigationItem(position);
            }

                @Override
                public void onPageScrolled(int arg0, float arg1, int arg2) {
            }
                @Override
                public void onPageScrollStateChanged(int arg0) {
            }
        });
        parser.execute();
        jmanager.execute();
        weather.execute();
        getApplicationContext();
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.main, menu);
    return true;
}
  

}

Это мой logcat

 FATAL EXCEPTION: main
java.lang.NullPointerException
at com.example.test23.VideosActivity.displayVideos(VideosActivity.java:54)
at com.example.test23.ParseManager.onPostExecute(ParseManager.java:52)
at com.example.test23.ParseManager.onPostExecute(ParseManager.java:1)
at android.os.AsyncTask.finish(AsyncTask.java:631)
at android.os.AsyncTask.access$600(AsyncTask.java:177)
at android.os.AsyncTask$InternalHandler.handleMessage(AsyncTask.java:644)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:5031)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:511)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:792)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:555)
at dalvik.system.NativeStart.main(Native Method)
  

ОБНОВЛЕНИЕ: это video_activity.xml

 <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical" >

        <ListView 
            android:id="@ id/videoslist"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" >
        </ListView>
     </LinearLayout>
  

Я просто хочу знать, что не так с моим приложением и почему я получаю значение context null, чтобы я мог продолжить работу с моим приложением, может быть, что-то со структурой, как она запрограммирована, я не знаю. Любая помощь приветствуется, если вам нужно больше кода, дайте мне знать.

Комментарии:

1. опубликуйте свой videos_activity.xml

2. @Rod_Algonquin только что опубликовал его!

Ответ №1:

Обязательно удалите «= getActivity()» из этой строки:

 Context context=getActivity();
  

Так что он просто читает:

 Context context;
  

Затем измените это:

 public View onCreateView(LayoutInflater inflater,ViewGroup container,Bundle        savedInstanceState){
   super.onCreateView(inflater, container, savedInstanceState);
   View view=inflater.inflate(R.layout.home_activity, container,false);
   newsList=(ListView)view.findViewById(R.id.newslist);
   setRetainInstance(true); 
   return view;
}
  

для этого:

 public View onCreateView(LayoutInflater inflater,ViewGroup container,Bundle        savedInstanceState){
   super.onCreateView(inflater, container, savedInstanceState);
   View view=inflater.inflate(R.layout.home_activity, container,false);
   newsList=(ListView)view.findViewById(R.id.newslist);
   context = getActivity();  // <--- add this line here
   setRetainInstance(true); 
   return view;
}
  

Комментарии:

1. Я сделал это, но я все еще получаю NPE, я протестировал его, но контекст по-прежнему равен нулю

2. Вы изменили все свои фрагменты, чтобы использовать описанный выше метод?

3. Да, во всех из них, но я все еще получаю NPE

4. Вы уверены, что itemAdapter — это то, что равно null, а не listVideo?

5. Лучшее, что я мог бы порекомендовать, это внутри displayVideos() после itemAdapter=new ItemAdapter(…) проверить, что itemAdapter != null, и установить адаптер только в том случае, если он не равен null.

Ответ №2:

вы можете использовать только getActivity() для установки контекста после onAttach того, как он был вызван, так что делайте

 context = getActivity() 
  

в вашем onCreateView

Комментарии:

1. покажите обновленный код и где в коде вы получаете исключение