我正在尝试保存RecyclerView的状态,以便即使在旋转时也能显示数据。我实际上通过保存LayoutManager的状态来解决这个问题。我在轮换时得到的错误是:
Process: com.example.android.guardiannewsapp, PID: 25829
java.lang.NullPointerException: Attempt to invoke virtual method 'android.os.Parcelable android.support.v7.widget.RecyclerView$LayoutManager.onSaveInstanceState()' on a null object reference
at com.example.android.guardiannewsapp.MainActivity.onSaveInstanceState(MainActivity.java:80)
我想我明白这个问题是什么,但不知道如何解决它。该错误指向此方法:
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
mListState = mLayoutManager.onSaveInstanceState();
outState.putParcelable(STATE_LIST, mListState);
}
和这一行
mListState = mLayoutManager.onSaveInstanceState();
这是我的MainActivity代码:
package com.example.android.guardiannewsapp;
import android.app.LoaderManager;
import android.app.LoaderManager.LoaderCallbacks;
import android.content.Context;
import android.content.Loader;
import android.graphics.Canvas;
import android.graphics.drawable.Drawable;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.os.Bundle;
import android.os.Parcelable;
import android.os.PersistableBundle;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.DefaultItemAnimator;
import android.support.v7.widget.DividerItemDecoration;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.View;
import android.widget.ProgressBar;
import android.widget.SearchView;
import android.widget.TextView;
import java.util.List;
public class MainActivity extends AppCompatActivity implements LoaderCallbacks<List<News>> {
//Constant value for the news loader ID. We can choose any integer. We do this if we
//are using multiple loaders. We aren't in this app but good practice.
private static final int EARTHQUAKE_LAODER_ID = 1;
RecyclerView recyclerView;
private TextView emptyView;
private ProgressBar progressBar;
private ConnectivityManager cm;
private TextView internetConnectionEmptyView;
private List<News> mNews;
private LoaderManager loaderManager;
private static final String STATE_LIST = "State Adapter Data";
RecyclerView.LayoutManager mLayoutManager;
private Parcelable mListState;
//Adapter for the list of news articles
private NewsAdapter mAdapter;
//User search term
private String userSearch = "";
//URL to query the Guardian dataset for search query
private String GUARDIAN_REQUEST_URL = "";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//Find a reference to the RecyclerView in the layout
recyclerView = findViewById(R.id.recyclerView);
//Find a reference to the empty view
emptyView = findViewById(R.id.empty_view);
//Find a reference to the progress bar and keep it hidden
progressBar = findViewById(R.id.progress_bar);
progressBar.setVisibility(View.GONE);
//Find a reference to the no internet connection message.
internetConnectionEmptyView = findViewById(R.id.no_internet_connection);
recyclerView.addItemDecoration(new DividerItemDecoration(getApplicationContext()));
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
mListState = mLayoutManager.onSaveInstanceState();
outState.putParcelable(STATE_LIST, mListState);
}
@Override
public void onRestoreInstanceState(Bundle outState) {
super.onRestoreInstanceState(outState);
if (outState != null) {
mListState = outState.getParcelable(STATE_LIST);
}
}
@Override
protected void onResume() {
super.onResume();
if (mListState != null) {
mLayoutManager.onRestoreInstanceState(mListState);
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
MenuInflater inflater = getMenuInflater();
inflater.inflate(R.menu.options_menu, menu);
// Associate searchable configuration with the SearchView
final SearchView searchView =
(SearchView) menu.findItem(R.id.search).getActionView();
searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
@Override
public boolean onQueryTextSubmit(String query) {
userSearch = query;
cm = (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
//Initialize the loader. Pass in the int ID constant defined above and pass in null
//for the bundle. Pass in this activity for the LoaderCallbacks parameter (which is valid
//because this activity implements the LoaderCallbacks interace).
if (activeNetwork != null && activeNetwork.isConnected()) {
// Get a reference to the LoaderManager, in order to interact with loaders.
loaderManager = getLoaderManager();
//Initialize the loader. Pass in the int ID constant defined above and pass in null
//for the bundle. Pass in the activity for the LoaderCallbacks parameter (which is valid
//because this activity implements the LoaderCallbacks interface.)
loaderManager.initLoader(EARTHQUAKE_LAODER_ID, null, MainActivity.this);
loaderManager.restartLoader(EARTHQUAKE_LAODER_ID, null, MainActivity.this);
} else {
//If there is no network connection, hide the loading indicator and show the
//no internet connection message.
internetConnectionEmptyView.setVisibility(View.VISIBLE);
progressBar.setVisibility(View.INVISIBLE);
}
searchView.clearFocus();
return true;
}
@Override
public boolean onQueryTextChange(String newText) {
return false;
}
});{
}
return true;
}
//When the LoaderManager determines that the loader with our specified ID isn't running to
//create a new one.
@Override
public Loader<List<News>> onCreateLoader(int i, Bundle bundle) {
GUARDIAN_REQUEST_URL = "http://content.guardianapis.com/search?show-fields=thumbnail&q="+userSearch+
"&api-key=23a6ee65-f55d-452f-a073-0bc71e36bb8b";
return new NewsLoader(this, GUARDIAN_REQUEST_URL);
}
@Override
public void onLoadFinished(Loader<List<News>> loader, List<News> news) {
mNews = news;
//If the list is empty, the app will show the emptyView message
if (news.isEmpty()){
emptyView.setVisibility(View.VISIBLE);
recyclerView.setVisibility(View.INVISIBLE);
progressBar.setVisibility(View.INVISIBLE);
} else {
//Once the view populates, hide the progress bar.
progressBar.setVisibility(View.INVISIBLE);
//Create a new adapter that takes an empty list of news articles as input
mAdapter = new NewsAdapter(this, news);
//Create a new adapter that takes an empty list of news articles as input
mAdapter = new NewsAdapter(this, news);
//Set the adapter on the RecyclerView so the list can be populated in the user interface.
mLayoutManager = new LinearLayoutManager(getApplicationContext());
recyclerView.setLayoutManager(mLayoutManager);
recyclerView.setItemAnimator(new DefaultItemAnimator());
recyclerView.setAdapter(mAdapter);
}
}
@Override
public void onLoaderReset(Loader<List<News>> loader) {
}
public class DividerItemDecoration extends RecyclerView.ItemDecoration {
private Drawable mDivider;
public DividerItemDecoration(Context context){
mDivider = context.getResources().getDrawable(R.drawable.horizontal_line);
}
@Override
public void onDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) {
int dividerLeft = parent.getPaddingLeft();
int dividerRight = parent.getWidth() - parent.getPaddingRight();
int childCount = parent.getChildCount();
for (int i = 0; i < childCount - 1; i++) {
View child = parent.getChildAt(i);
RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();
int dividerTop = child.getBottom() + params.bottomMargin;
int dividerBottom = dividerTop + mDivider.getIntrinsicHeight();
mDivider.setBounds(dividerLeft, dividerTop, dividerRight, dividerBottom);
mDivider.draw(c);
}
}
}
}
答案 0 :(得分:1)
看起来mLayoutManager
尚未设置。由于它在onLoadFinished()
中设置,因此onLoadFinished()
未运行或news.isEmpty()
为真,因此仍未设置mLayoutManager
。
应该没问题,所以只需在mLayoutManager
中设置onCreate()
即可。那应该可以解决问题。您还可以在mLayoutManager
中使用它之前检查onSaveInstanceState()
是否为空。