Java.lang.NullPointerException в файле android.support.v7.widget.RecyclerView.onMeasure

Я пытаюсь использовать RecyclerView с RecyclerView.Adapter, но здесь что-то не так. Я размещаю свой код ниже:

Разметка:

   <android.support.v7.widget.RecyclerView
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/topic_view"
    android:scrollbars="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

topic_tile.xml:

<?xml version="1.0" encoding="utf-8"?>

<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="@dimen/tile_height">

    <com.makeramen.RoundedImageView
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/avatar"
        android:padding="16dp"
        app:riv_corner_radius="72dp"
        android:layout_height="72dp"
        android:layout_width="72dp"
        />

    <LinearLayout
        android:id="@+id/text_layout"
        android:orientation="vertical"
        android:paddingTop="@dimen/text_padding_top_and_bottom"
        android:paddingBottom="@dimen/text_padding_top_and_bottom"
        android:paddingRight="16dp"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <TextView
            android:id="@+id/title"
            android:textSize="@dimen/primary_font"
            android:paddingLeft="@dimen/text_padding_left"
            android:textColor="#000000"
            android:text="Title"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>
        <TextView
            android:id="@+id/author"
            android:textSize="@dimen/secondary_font"
            android:paddingLeft="@dimen/text_padding_left"
            android:text="author"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>
    </LinearLayout>
</RelativeLayout>

Здесь находится onCreate()

public class TitleListActivity extends ActionBarActivity {

    private RecyclerView topic_view;
    private RecyclerView.LayoutManager mLayoutManager;
    private TitlelistAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_title_list);

        .....

        topic_view = (RecyclerView)findViewById(R.id.topic_view);
        adapter = new TitlelistAdapter(topicList);

        topic_view.setHasFixedSize(true);
        mLayoutManager = new LinearLayoutManager(TitleListActivity.this);
        topic_view.setLayoutManager(mLayoutManager);
        topic_view.setItemAnimator(new DefaultItemAnimator());

        topic_view.setAdapter(adapter);

Адаптер:

public class TitlelistAdapter extends RecyclerView.Adapter<TitlelistAdapter.ViewHolder> {

    public List<Topic> topicList;

    public static class ViewHolder extends RecyclerView.ViewHolder{

        TextView title;
        TextView author;
        RoundedImageView avatar;

        public ViewHolder(View itemView) {
            super(itemView);

            title = (TextView)itemView.findViewById(R.id.title);
            author = (TextView)itemView.findViewById(R.id.author);
            avatar = (RoundedImageView)itemView.findViewById(R.id.avatar);
        }
    }

    public TitlelistAdapter(List<Topic> topicList){
        this.topicList = topicList;
    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {

        View itemView = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.topic_tile,viewGroup,false);
        return new ViewHolder(itemView);
    }

    @Override
    public void onBindViewHolder(ViewHolder viewHolder, int i) {
        viewHolder.title.setText(topicList.get(i).title);
        viewHolder.author.setText(topicList.get(i).member.username);
    }

    @Override
    public long getItemId(int position) {
        return super.getItemId(position);
    }

    @Override
    public int getItemCount() {
        return topicList.size();
    }
}

Вот исключение:

java.lang.NullPointerException
        at android.support.v7.widget.RecyclerView.onMeasure(RecyclerView.java:1694)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
        at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1404)
        at android.widget.LinearLayout.measureVertical(LinearLayout.java:695)
        at android.widget.LinearLayout.onMeasure(LinearLayout.java:588)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
        at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
        at android.support.v7.internal.widget.ActionBarOverlayLayout.onMeasure(ActionBarOverlayLayout.java:453)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
        at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
        at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1404)
        at android.widget.LinearLayout.measureVertical(LinearLayout.java:695)
        at android.widget.LinearLayout.onMeasure(LinearLayout.java:588)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
        at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
        at com.android.internal.policy.impl.PhoneWindow$DecorView.onMeasure(PhoneWindow.java:2291)
        at android.view.View.measure(View.java:16497)
        at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:1912)
        at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1109)
        at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1291)
        at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:996)
        at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5600)
        at android.view.Choreographer$CallbackRecord.run(Choreographer.java:761)
        at android.view.Choreographer.doCallbacks(Choreographer.java:574)
        at android.view.Choreographer.doFrame(Choreographer.java:544)
        at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:747)
        at android.os.Handler.handleCallback(Handler.java:733)
        at android.os.Handler.dispatchMessage(Handler.java:95)
        at android.os.Looper.loop(Looper.java:136)
        at android.app.ActivityThread.main(ActivityThread.java:5001)
        at java.lang.reflect.Method.invokeNative(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:515)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:785)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:601)
        at dalvik.system.NativeStart.main(Native Method)

Это что-то, что я не запустил, или что-то, что я должен написать больше?

Ответ 1

Я получил эту ошибку, когда у моего RecyclerView не было LayoutManager. При добавлении этого кода исправлена ​​проблема:

recyclerView.setLayoutManager(new LinearLayoutManager(recyclerView.getContext()));

Ответ 2

Ваш xml показывает, что у вас есть два пространства имен android, которые на самом деле должны дать вам ошибку, потому что в android вам разрешено использовать пространство имен только один раз. Удалите linearLayout из основного макета, поскольку он кажется ненужным. Проверьте свой собственный макет Row i.e topic_tile, а также для любых подобных ошибок

<android.support.v7.widget.RecyclerView 
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/topic_view"
    android:scrollbars="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

ИЗМЕНИТЬ:

in onCreate()

private RecyclerView.LayoutManager mLayoutManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        .
        . 
        .
        mLayoutManager = new LinearLayoutManager(this);
        recycleView.setLayoutManager(mLayoutManager);
    }

Ответ 3

Если вы сталкиваетесь с той же проблемой для RecycleView внутри фрагмента, попробуйте использовать этот код.

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    // Inflate the layout for this fragment
    View rootView = inflater.inflate(R.layout.fragment_latest, container, false);

    // get recycler view
    mRecyclerView = (RecyclerView) rootView.findViewById(R.id.novel_item_list);

    //mRecyclerView.setHasFixedSize(true);

    // use a linear layout manager
    mLayoutManager = new LinearLayoutManager(getActivity());
    mRecyclerView.setLayoutManager(mLayoutManager);


    String[] abc = {"hi","how are you","this is recycler"};
    // specify an adapter (see also next example)
    mAdapter = new RecyclerViewAdapter(abc);
    mRecyclerView.setAdapter(mAdapter);
    return rootView;
}

Ответ 4

Вы должны установить LayoutManager для RecycleView:

В настоящее время андроид поддерживает 3 вида макетов.

Если вы хотите, чтобы это выглядело нормально ListView, используйте LinearLayoutManager.

Если вы хотите, чтобы это выглядело gridView, используйте GridLayoutManager.

Если вы хотите, чтобы это выглядело в шахматном порядке, используйте StaggeredGridLayoutManager

Конструкторы для трех видов макетов выше

LinearLayoutManager linearLayoutManager = new LinearLayoutManager(getActivity());

GridLayoutManager gridLayoutManager = new GridLayoutManager(getActivity(), 2);

StaggeredGridLayoutManager staggeredGridLayoutManager = new StaggeredGridLayoutManager(spanCount, StaggeredGridLayoutManager.HORIZONTAL);

Затем вы установите LinearLayout в RecycleView (в этом случае я использовал LinearLayoutManager

recyclerView = (RecyclerView) view.findViewById(R.id.saved_recycle_view);
recyclerView.setHasFixedSize(true);
LinearLayoutManager linearLayoutManager = new LinearLayoutManager(context);
recyclerView.setLayoutManager(linearLayoutManager);

В любом случае, используйте ОттоБус Библиотека, чтобы передать события из Адаптера в Активность/Фрагмент, содержащий RecycleView. Удачи!

Ответ 5

Для тех, кто пытается использовать ViewPager для вкладок, а внутри их вкладок есть RecyclerView, который вызывает эту сумасшедшую ошибку, я предлагаю отложить раздувание, создав еще один фрагмент для включения только RecylerView.

Этот xml файл является одним из моих вкладок, которые содержат компоновку фрейма вместо recycler.

shop_most_views_fragment.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">

<FrameLayout
    android:id="@+id/most_view_item_frameLayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#f3f3f3" />
</LinearLayout>

его класс java

public class MostViews extends Fragment {
private String title;
private int page;
View convertedView;
RecyclerView grids;

public static MostViews newInstance(int page, String title) {
    MostViews fragmentFirst = new MostViews();
    Bundle args = new Bundle();
    args.putInt("someInt", page);
    args.putString("someTitle", title);
    fragmentFirst.setArguments(args);
    return fragmentFirst;
}

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    page = getArguments().getInt("someInt", 0);
    title = getArguments().getString("someTitle");
}

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    convertedView = inflater.inflate(R.layout.shop_most_views_fragment, container, false);
    transitMasonaryLayoutFragmentIntoFramLayout();
    return convertedView;
}

private void transitMasonaryLayoutFragmentIntoFramLayout() {
    MasanoryLayout masanoryLayout = new MasanoryLayout();
    FragmentManager fragmentManager = getActivity().getSupportFragmentManager();
    fragmentManager.beginTransaction()
            .add(R.id.most_view_item_frameLayout, masanoryLayout).commit();
}
}

поэтому у нас есть макет кадра, чтобы вставить в него наш фрагмент recylcer.

shop_most_view_masanory_fragment.xml

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

<android.support.v7.widget.RecyclerView
    android:id="@+id/mostViewsItem"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />
</LinearLayout>

его класс java

public class MasanoryLayout extends Fragment {
View convertedView;
RecyclerView grids;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    convertedView = inflater.inflate(R.layout.shop_most_view_masanory_fragment, container, false);
    grids = (RecyclerView) convertedView.findViewById(R.id.mostViewsItem);
    setLayoutManagerOnRecyclerView();
    MasonryAdapter adapter = new MasonryAdapter(initializeData());
    grids.setAdapter(adapter);
    return convertedView;
}

private void setLayoutManagerOnRecyclerView() {
    grids.setLayoutManager(new StaggeredGridLayoutManager(3, StaggeredGridLayoutManager.VERTICAL));
}

private List<StoreItem> initializeData() {
    List<StoreItem> storeItems = new ArrayList<>();
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.detailed_index_store));
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.clothstore));
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.detailed_index_store));
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.clothstore));
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.detailed_index_store));
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.detailed_index_store));
    storeItems.add(new StoreItem("احسان", "احسان", R.drawable.clothstore));

    return storeItems;
}
}

как вы видите, ваш FragmentPagerAdapter будет называть MostViews для раздувания, а MostViews сам позовет MasanoryLayout для раздувания. поэтому мы избежали контекста FragmentPagerAdapter;)