片段未附加到上下文

时间:2018-05-09 15:55:05

标签: android android-fragments

在工具栏的活动中,我得到了一个按钮,需要从该片段中的片段和更新列表中调用方法。现在是一个错误。 呼叫活动

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()){
        case  R.id.menu_sort:
            ListFragment listFragment = new ListFragment();
            listFragment.sortByPopularity();
            break;
    }
    return super.onOptionsItemSelected(item);
}

片段代码。我没有附加活动时发现错误。但没有上下文

public class ListFragment extends Fragment implements ListAdapter.ItemClickListener {

    /**
     * Needed
     */
    RecyclerView recyclerView;
    View view;
    List<BasePojo.Result> list;
    ListAdapter listAdapter;

    public ListFragment() {
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {

        /**
         * Main Initialization
         */
        view = inflater.inflate(R.layout.fragment_list, container, false);
        recyclerView = view.findViewById(R.id.recycler_list_detailed);
        recyclerView.setLayoutManager(new GridLayoutManager(getActivity(), 2));
        list = new ArrayList<>();
        listAdapter = new ListAdapter(list, setOnItemClickCallback());
        recyclerView.setAdapter(listAdapter);

        RetrofitClient.getApiService().getPhotosList(getString(R.string.api_key)).enqueue(new Callback<BasePojo>() {
            @Override
            public void onResponse(Call<BasePojo> call, Response<BasePojo> response) {
                BasePojo basePojo = response.body();
                list.addAll(basePojo.getResults());
                recyclerView.getAdapter().notifyDataSetChanged();
            }

            @Override
            public void onFailure(Call<BasePojo> call, Throwable t) {
                Log.d("tag", "Response failed" + t.toString());

            }
        });


        return view;
    }

    @Override
    public void onItemClick(View view, int position) {
        Log.v("in on click", "value " + position);

    }

    private OnItemClickListener.OnItemClickCallback setOnItemClickCallback() {
        OnItemClickListener.OnItemClickCallback onItemClickCallback = new OnItemClickListener.OnItemClickCallback() {
            @Override
            public void onItemClicked(View view, int position) {
                BasePojo.Result itemClicked = list.get(position);
                Bundle bundle = new Bundle();
                bundle.putString("title", itemClicked.getOriginalTitle());
                bundle.putString("overview", itemClicked.getOverview());
                bundle.putString("release_date", itemClicked.getReleaseDate());
                bundle.putString("vote_average", itemClicked.getVoteAverage().toString());
                bundle.putString("poster_path", itemClicked.getPosterPath());
                DetailedFragment detailedFragment = new DetailedFragment();
                detailedFragment.setArguments(bundle);
                FragmentManager manager = getActivity().getSupportFragmentManager();
                FragmentTransaction transaction = manager.beginTransaction();
                transaction.replace(R.id.main_frame_list, detailedFragment);
                Log.d("tag", "title is 111 " + bundle.get("title"));

                transaction.commit();
            }

        };
        return onItemClickCallback;
    }

    @Override
    public void onAttachFragment(Fragment childFragment) {
        super.onAttachFragment(childFragment);

    }

    public void sortByPopularity() {
        RetrofitClient.getApiService().getPopularList(getString(R.string.api_key)).enqueue(new Callback<BasePojo>() {
            @Override
            public void onResponse(Call<BasePojo> call, Response<BasePojo> response) {
                BasePojo basePojo = response.body();
                list.addAll(basePojo.getResults());
                recyclerView.getAdapter().notifyDataSetChanged();
            }

            @Override
            public void onFailure(Call<BasePojo> call, Throwable t) {
                Log.d("tag", "Response failed" + t.toString());

            }
        }); }

}

这是一个错误

05-09 12:48:26.915 5775-5775/com.borisruzanov.popularmovies E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.borisruzanov.popularmovies, PID: 5775
java.lang.IllegalStateException: Fragment ListFragment{6dbd6de} not attached to a context.
    at android.support.v4.app.Fragment.requireContext(Fragment.java:614)
    at android.support.v4.app.Fragment.getResources(Fragment.java:678)
    at android.support.v4.app.Fragment.getString(Fragment.java:700)
    at com.borisruzanov.popularmovies.ListFragment.sortByPopularity(ListFragment.java:110)
    at com.borisruzanov.popularmovies.MainActivity.onOptionsItemSelected(MainActivity.java:47)
    at android.app.Activity.onMenuItemSelected(Activity.java:3204)
    at android.support.v4.app.FragmentActivity.onMenuItemSelected(FragmentActivity.java:407)
    at android.support.v7.app.AppCompatActivity.onMenuItemSelected(AppCompatActivity.java:195)
    at android.support.v7.view.WindowCallbackWrapper.onMenuItemSelected(WindowCallbackWrapper.java:108)
    at android.support.v7.view.WindowCallbackWrapper.onMenuItemSelected(WindowCallbackWrapper.java:108)
    at android.support.v7.app.ToolbarActionBar$2.onMenuItemClick(ToolbarActionBar.java:63)
    at android.support.v7.widget.Toolbar$1.onMenuItemClick(Toolbar.java:203)
    at android.support.v7.widget.ActionMenuView$MenuBuilderCallback.onMenuItemSelected(ActionMenuView.java:780)
    at android.support.v7.view.menu.MenuBuilder.dispatchMenuItemSelected(MenuBuilder.java:822)
    at android.support.v7.view.menu.MenuItemImpl.invoke(MenuItemImpl.java:171)
    at android.support.v7.view.menu.MenuBuilder.performItemAction(MenuBuilder.java:973)
    at android.support.v7.view.menu.MenuBuilder.performItemAction(MenuBuilder.java:963)
    at android.support.v7.widget.ActionMenuView.invokeItem(ActionMenuView.java:624)
    at android.support.v7.view.menu.ActionMenuItemView.onClick(ActionMenuItemView.java:150)
    at android.view.View.performClick(View.java:5610)
    at android.view.View$PerformClick.run(View.java:22265)
    at android.os.Handler.handleCallback(Handler.java:751)
    at android.os.Handler.dispatchMessage(Handler.java:95)
    at android.os.Looper.loop(Looper.java:154)
    at android.app.ActivityThread.main(ActivityThread.java:6077)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:866)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:756)

非常感谢你的时间和帮助。如果我的问题看起来不太好请注意,我将教会如何更好地提出问题

10 个答案:

答案 0 :(得分:44)

就我而言,当我打电话给getString()

时发生了此问题

将此调用更改为getActivity().getString()解决了该问题。

答案 1 :(得分:6)

科特琳:

我的问题出在getString()

将其更改为context.getString()即可解决

答案 2 :(得分:5)

创建片段实例是不够的。
它需要通过交易附加到活动:

getFragmentManager()
    .beginTransaction()
    .replace(R.id.container_layout, fragment)
    .commit();

成功提交后,调用片段中的onAttach方法,创建视图,然后您可以与其视图进行交互。

在您的情况下,创建片段实例并将其附加到活动onCreate中,然后在点击事件中稍后调用sortByPopularity

详细了解片段生命周期:https://developer.android.com/guide/components/fragments

答案 3 :(得分:2)

如果您使用的是 CountDownTimer ,则可能会由于错误原因而导致在完成计时器之前分离片段。如果您要在 onFinish 回调中执行ui更改,则应检查上下文是否为null或如下所示;

    timer = object : CountDownTimer(startTimeInMillis, 1000) {
        override fun onTick(millisUntilFinished: Long) {

        }

        override fun onFinish() {
            context?.let {
              //perform ui changes here 
            }
        }
    }
    timer?.start()

或者您应该在分离片段之前取消计时器,如下所示;

override fun onDestroy() {
    super.onDestroy()
    timer?.cancel()
}

答案 4 :(得分:1)

如果片段未显示(未添加)或被移除,则为 context == null。在这种情况下,获取资源将导致此异常。 getString(R.string.some_string) 需要 context 并且崩溃。

您可以检查片段是否存在:

if (isAdded) {
    // Print getString(R.string.some_string).
}

但是,即使片段已发布,您也可能需要打印字符串,例如,在 LogCat、分析中或向服务器发送请求。在这种情况下,您需要一个应用程序上下文来获取字符串资源。

class MyApplication : Application() {

    override fun onCreate() {
        super.onCreate()

        instance = this
    }


    companion object {
        lateinit var instance: MyApplication private set
    }
}

object Strings {
    fun get(@StringRes stringRes: Int, vararg formatArgs: Any = emptyArray()): String {
        return instance.getString(stringRes, *formatArgs)
    }
}

然后在 AndroidManifest 中设置 MyApplication 并使用:Strings.get(R.string.some_string)

答案 5 :(得分:0)

使用commit()无法解决问题,我们应该尝试在Fragment的源代码中找到解决方案。

因此,从您提供的错误堆栈中考虑,Fragment中的requireContext()为:

    public final Context requireContext() {
        Context context = getContext();
        if (context == null) {
            throw new IllegalStateException("Fragment " + this + " not attached to a context.");
        }
        return context;
    }

这意味着系统将从Context中检查getContext(),如果为空,则将引发异常。

因此,为避免此问题,我们可以在开展业务之前检查getContext()的结果。

答案 6 :(得分:0)

我知道这是一篇过时的文章,但是我只是想出了你能做些什么。确实,创建片段实例是不够的,它需要通过事务附加到Activity。但是,您可以首先添加两个片段,并将它们与片段管理器分离。这样,它们在fragmentManager中都是“活跃的”,您以后可以根据需要在这些碎片上调用attach和detach。

.add(container, fragment1).detach(fragment1).add(container, fragment2).commit();
.
.
.
.
.
ft.detach(fragment2)
ft.attach(fragment1

答案 7 :(得分:0)

科特琳:使用延迟初始化

override val contentMessage by lazy {
     getString(R.string.message)
}

答案 8 :(得分:0)

在这种情况下,请验证您是否没有任何依赖于上下文的类级别属性,因为片段未提交,它将没有上下文,我们最终可能会遇到此异常。

答案 9 :(得分:0)

该断言可以发生在您通过 requireContext() 调用从 Fragment 访问 Android 上下文的任何地方。在您使用 requireContext() 之前,请仔细查看呼叫站点。我只在我确定片段当时将附加到 Activity 或用例非常重要以至于使用此断言崩溃比任何其他操作过程更好时才使用 requireContext()

如果由于任何原因片段可能没有附加并且您可以通过避免或提前返回在调用站点自己处理它,那么更好的主意是对 getContext() 的返回进行空检查,然后才继续前进。

空检查的典型 Kotlin 代码如下所示:

fun myFragmentFunction(){
    val context = getContext() ?: return // early return using Elvis operator
    context.whatever() // guaranteed non-null context at this point
}