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

Posted

技术标签:

【中文标题】使用 Fragment 内的方法更改 TextView 值 (NullPointerException)【英文标题】:Change TextView values with method inside Fragment (NullPointerException) 【发布时间】:2020-03-08 15:08:49 【问题描述】:

我正在尝试制作一个应用程序(用于学习目的),其中有一些带有可滚动内容的标签。 我有一个可折叠的应用栏,其中一个 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_
    android:layout_>

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

        android:layout_
        android:layout_>

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

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


            <ImageView
                android:id="@+id/imageView7"
                android:layout_
                android:layout_
                app:srcCompat="@drawable/strawberry" />

            <androidx.appcompat.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_
                android:layout_></androidx.appcompat.widget.Toolbar>

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

        <com.google.android.material.tabs.TabLayout
            android:id="@+id/tabLayout"
            android:layout_
            android:layout_
            android:visibility="visible" />

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


    <androidx.viewpager.widget.ViewPager
        android:id="@+id/viewPager"
        android:layout_
        android:layout_
        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_
    android:layout_>

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

这是它应该如何工作的(从之前每个选项卡都有一个片段类和一个布局文件):

【问题讨论】:

在你得到 NullPointerException 的地方添加堆栈跟踪 【参考方案1】:

问题是创建的 TabFragment 实例尚未附加到 Activity,因此此时 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 Reference看到一个很好的例子。

【讨论】:

以上是关于使用 Fragment 内的方法更改 TextView 值 (NullPointerException)的主要内容,如果未能解决你的问题,请参考以下文章

检测片段内的 ViewPager 选项卡更改

无法在 Fragment 的 onCreateView 方法中更改 TextView 的文本

Android getParentFragment() 在 Fragment 内的 ViewPager 中返回 null

将Sqlite数据放入两个不同Fragment内的两个ListView

UITextView 确实完成了更改选择

单击不在 Fragment 内的 RecyclerView 项目上工作