gpt4 book ai didi

java - Android findFragmentByTag调试时返回null

转载 作者:塔克拉玛干 更新时间:2023-11-02 19:21:11 26 4
gpt4 key购买 nike

背景:

我正在编写一个 android 应用程序,其中包含一个可以由多个 fragment 之一填充的 Activity 。通常,我需要将一组数据传递给 Activity fragment ,以便它可以相应地更新 UI。

为了获取当前 fragment ,我调用了 getfragmentManager().findFragmentByTag() 并将返回的 fragment 转换为我的自定义 fragment 类。

问题:

上述方法通常都可以正常工作。但是,findFragmentByTag() 偶尔会返回 null。经过进一步调查,我得出结论,这只会在我从 Android studio 运行或调试时发生(即使这样也不会每次都发生)。

相关代码:

protected void onCreate(Bundle savedInstanceState){

//Do lots of stuff
currentFragmentTag = "";
getFragmentManager().addOnBackStackChangedListener(
new FragmentManager.OnBackStackChangedListener() {
public void onBackStackChanged() {
if (getFragmentManager().getBackStackEntryCount() > 0) {
currentFragmentTag = getFragmentManager().getBackStackEntryAt(getFragmentManager().getBackStackEntryCount() - 1).getName();
}

}
});
init();

//Do some more stuff

//this should always be the case
if (currentFragmentTag.equals(LOGIN_FRAGMENT_TAG)) {
((LoginFragment) getFragmentManager().findFragmentByTag(currentFragmentTag)).beginLogin();
}

}


private void init(){
//changed to reflect George Mulligan's advice
Fragment currentFrag = getFragmentManager().findFragmentByTag(LOGIN_FRAGMENT_TAG);
if(currentFrag == null) {
currentFragmentTag = LOGIN_FRAGMENT_TAG;
getFragmentManager().beginTransaction()
.add(R.id.container, loginFrag, LOGIN_FRAGMENT_TAG)
.addToBackStack(LOGIN_FRAGMENT_TAG)
.commit();
getFragmentManager().executePendingTransactions();

}
}

public void updateFragment(){
Bundle dataBundle = new Bundle();
//put stuff in dataBundle

if (currentFragmentTag.equals(LOGIN_FRAGMENT_TAG)) {
LoginFragment currentFrag = (LoginFragment) getFragmentManager().findFragmentByTag(currentFragmentTag);
if (currentFrag != null) {
currentFrag.passBundleToFragment(dataBundle);
Log.d(TAG, "Fragment returned is valid.");
} else {
Log.d(TAG, "Fragment returned is null.");
}
}
//else if a different fragment is active then update it in the same way
}

//Manually open the loginFragment. This can be called from other fragments. My problem always occurs before this is called however.
@Override
public void openLoginScreen() {
if(/*some conditions*/) {
LoginFragment loginFrag = LoginFragment.newInstance();
getFragmentManager().beginTransaction()
.replace(R.id.container, loginFrag, LOGIN_FRAGMENT_TAG)
.addToBackStack(LOGIN_FRAGMENT_TAG)
.commit();
getFragmentManager().executePendingTransactions();
currentFragmentTag = LOGIN_FRAGMENT_TAG;
updateFragment();
}
}

通常,我的 logcat 看起来像这样:

Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
...etc.

但时不时地,只有当我从 Android Studio 启动应用程序时,我才会得到类似这样的信息:

Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is valid
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is null
Fragment returned is valid
Fragment returned is valid

这到底是怎么回事?

更新:

通过单击应用切换按钮,关闭我的应用并立即重新启动它,我已经能够在与 Android Studio 断开连接时重现此错误。如果我足够快地执行此操作,它永远不会像我上面描述的那样运行。

在记录更多日志并追查其他问题后,我发现在这些特定情况下,onCreate() 被调用了两次。

我的应用程序设计为仅在横向模式下运行,部分原因是为了避免重新创建 Activity 时出现问题。然而,当应用程序关闭并快速重新启动时,Android 似乎永远不会在我的应用程序再次启动之前完成必要的主屏幕纵向模式旋转。我的假设是,这会导致操作系统在应用程序运行后旋转回横向,从而重新启动它。

除了无法解释为什么 findFragmentByTag() 有时会返回 null 之外,所有这些都很好而且很花哨。

我的 Activity 类中的每个对象都应该重新创建,对吗?那么 FragmentManager 不应该也重新初始化吗?或者 getFragmentManager() 是对 Activity 本身之外的东西的静态引用?

第二次更新:

我尝试了 George 的想法,即在我调用 beginTransaction() 之前检查 fragment 是否已经添加,虽然它没有解决问题,但我在调试时注意到了一些奇怪的事情:

我在 Log.d(TAG, "Fragment returned is null."); 处设置了一个断点。关闭应用程序并快速重新启动它可确保达到我上面提到的此代码。然后,如果我通过在 Evaluate Expression 窗口中调用 getFragmentManager() 查看 Fragment Manager,我注意到已经添加了一个“登录 fragment ”,但它没有关联的标签.

Log.d(TAG, "Fragment returned is valid."); 处设置断点,但是,在同一应用程序 session 中,显示添加了 LoginFragment一个符合预期的标签。

在我的代码中没有任何意义可以在不设置标签的情况下添加 fragment 。这是否与正在重新创建的 Activity 和 fragment 管理器丢失标签有关,即使它持有 fragment 本身?

最佳答案

有几点需要注意。在 updateFragment 中,你真的应该使用 .equals 来比较你的字符串,而不是引用比较 == 所以 LOGIN_FRAGMENT_TAG.equals(currentFragmentTag)作为最佳实践并避免混淆。

此外,FragmentManager 会记住您在方向更改时添加到其中的 fragment 。因此,在您提到的您两次进入 onCreate 的示例中,您可能会在返回堆栈上有两个 LoginFrag 实例。

您可以通过在 updateFragment 方法中执行相同的查找来避免这种情况,并且仅在找不到 fragment 时才添加该 fragment 。

LoginFragment currentFrag = (LoginFragment) getFragmentManager()
.findFragmentByTag(LOGIN_FRAGMENT_TAG);

if(currentFrag == null) {
currentFrag = getFragmentManager().beginTransaction()
.replace(R.id.container, loginFrag, LOGIN_FRAGMENT_TAG)
.addToBackStack(LOGIN_FRAGMENT_TAG)
.commit();

//executePendingTransactions() usually isn't necessary...
getFragmentManager().executePendingTransactions();
}

我猜测此 Activity 中使用了多个 fragment ,因为您对字符串 currentFragmentTag 感到厌烦,但如果没有,那么您最好只保留对 LoginFragment< 的引用 作为类的一个字段,这样您就可以避免在 updateFragment 方法中额外查找它。

除此之外,我尝试重现您的错误但我不能,所以错误可能在您未显示的代码中的其他地方。

关于java - Android findFragmentByTag调试时返回null,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34515801/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com