使用 Fragment 中的方法更改 TextView 值 (NullPointerException)

Change TextView values with method inside Fragment (NullPointerException)

提问人:Laila Campos 提问时间:11/13/2019 更新时间:11/13/2019 访问量:188

问:

我正在尝试制作一个应用程序(用于学习目的),其中有几个带有可滚动内容的选项卡。 我有一个折叠的应用栏,其中有一个用于主布局的 ViewPager,以及三个相同的布局,其中 NestedScrollView 内有一个 TextView,唯一的区别是文本顶部的图像。我还有三个具有相同代码的片段类。一切都很好。

但是由于我在每个选项卡中使用相同的布局,因此我认为我可以只拥有每个选项卡中的一个,然后以某种方式更改适配器中每个选项卡的内容,而不是每个选项卡的 xml 文件和 java 类。

因此,我尝试在我的 tabFragment 类中创建一个方法来执行此操作。但我似乎找不到修改 TextView 的方法。我每次都收到 NullPointerException。

我错过了什么? 我的方法错了吗?如果是这样,什么会更好?我该如何解决这个问题?

提前感谢您:)


这是我现在的片段类(唯一剩下的):

import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import androidx.fragment.app.Fragment;
import com.example.tablayoutwithscrollview.R;

public class TabFragment extends Fragment {

    private TextView textView;
    private int text, image;

    public TabFragment(){}

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
    {
        View view = inflater.inflate(R.layout.tab1, container, false);
        //textView = view.findViewById(R.id.textView);
        return view;
    }

    @Override
    public void onViewCreated(View view, Bundle savedInstanceState)
    {
        super.onViewCreated(view, savedInstanceState);
        textView = getView().findViewById(R.id.textView);
    }

    public void setTabContent(int text, int image)
    {
        textView.setText(text);
        textView.setCompoundDrawablesWithIntrinsicBounds(0,image,0,0);
    }
}

我的适配器类:

import android.content.Context;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentPagerAdapter;

import com.example.tablayoutwithscrollview.Tabs.TabFragment;

public class MyAdapter extends FragmentPagerAdapter {

    private static int tabCount;
    Context context;

    public MyAdapter(FragmentManager fm, int tabCount, Context context){
        super(fm);
        this.tabCount = tabCount;
        this.context = context;
    }

    @Override
    public Fragment getItem(int position) {

        TabFragment fragment;
        switch (position)
        {
            case 0:
                fragment = new TabFragment();
                fragment.setTabContent(R.string.lorem_ipsum, R.drawable.jackfruit);
                return fragment;
            case 1:
                fragment = new TabFragment();
                fragment.setTabContent(R.string.lorem_ipsum, R.drawable.guava);
                return fragment;
            case 2:
                fragment = new TabFragment();
                fragment.setTabContent(R.string.lorem_ipsum, R.drawable.pomegranate);
                return fragment;
            default:
                return null;
        }
    }

    @Override
    public CharSequence getPageTitle(int position)
    {
        switch (position)
        {
            case 0:
                return context.getString(R.string.tab1_title);
            case 1:
                return context.getString(R.string.tab2_title);
            case 2:
                return context.getString(R.string.tab3_title);
            default:
                return null;
        }
    }

    @Override
    public int getCount() {
        return tabCount;
    }
}

主要活动:

package com.example.tablayoutwithscrollview;

import androidx.appcompat.app.AppCompatActivity;
import androidx.core.widget.NestedScrollView;
import androidx.viewpager.widget.ViewPager;

import android.os.Bundle;

import com.google.android.material.tabs.TabLayout;

public class MainActivity extends AppCompatActivity {

    private TabLayout tabLayout;
    private ViewPager viewPager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tabLayout = findViewById(R.id.tabLayout);
        viewPager = findViewById(R.id.viewPager);
        populatesTabLayout(tabLayout);
        MyAdapter myAdapter = new MyAdapter(getSupportFragmentManager(), tabLayout.getTabCount(), this);
        viewPager.setAdapter(myAdapter);

        tabLayout.setupWithViewPager(viewPager);
    }

    private void populatesTabLayout(TabLayout tabLayout)
    {
        for (int i = 0; i<3; i++)
        {
            tabLayout.addTab(tabLayout.newTab());
        }
    }
}

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"

    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/coordLayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.google.android.material.appbar.AppBarLayout
        android:id="@+id/appBarLayout"

        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <com.google.android.material.appbar.CollapsingToolbarLayout
            android:id="@+id/collapseToolBar"
            android:layout_width="match_parent"
            android:layout_height="match_parent"

            android:fitsSystemWindows="true"
            app:contentScrim="?attr/colorPrimary"
            app:layout_scrollFlags="scroll|exitUntilCollapsed"
            app:toolbarId="@+id/toolbar">


            <ImageView
                android:id="@+id/imageView7"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:srcCompat="@drawable/strawberry" />

            <androidx.appcompat.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"></androidx.appcompat.widget.Toolbar>

        </com.google.android.material.appbar.CollapsingToolbarLayout>

        <com.google.android.material.tabs.TabLayout
            android:id="@+id/tabLayout"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:visibility="visible" />

    </com.google.android.material.appbar.AppBarLayout>


    <androidx.viewpager.widget.ViewPager
        android:id="@+id/viewPager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />


</androidx.coordinatorlayout.widget.CoordinatorLayout>

还有我的选项卡布局:

<?xml version="1.0" encoding="utf-8"?>
<androidx.core.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/tabNestedScrollView"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/textView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:drawableTop="@drawable/jackfruit"
        android:text="@string/lorem_ipsum" />
</androidx.core.widget.NestedScrollView>

以下是它应该如何工作(从之前每个选项卡都有一个片段类和一个布局文件开始):enter image description here enter image description here enter image description here

android-layout android-fragments nullpointerexception findViewByID

评论

0赞 Rishabh Dhawan 11/13/2019
添加堆栈跟踪,其中获取 NullPointerException

答:

2赞 Szilárd Gerlei 11/13/2019 #1

问题在于创建的 TabFragment 实例尚未附加到活动,因此此时 textView 不会膨胀。

应将文本作为 MyAdapter#getItem(..) 方法中的参数传递给 TabFragment 实例。

...    
case 0:
   fragment = new TabFragment();

   Bundle args = new Bundle();
   args.putString("key", textString);
   fragment.setArguments(args);
   return fragment
...

之后,在 onViewCreated(...) 方法中设置 TextView 实例的文本:

@Override
public void onViewCreated(View view, Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);
    textView = getView().findViewById(R.id.textView);
    textView.setText(getArguments().getString("key"));
}

或者,您可以在 Android 参考中看到一个很好的示例。