关于Activity的onNewIntent方法

本文通过实验探讨了Android中Activity的不同启动模式(standard、singleTop、singleTask和singleInstance),并详细解释了onNewIntent方法的调用时机及其应用场景。

前言

onNewIntent方法想必大家都知道,是和Activity的启动模式结合起来使用的,可以这个方法具体什么情况下被调用,如何使用你清楚了吗?今天就来一探究竟,扫清疑惑。

实验

实验一

我们定义两个Activity,HomeActivity和SecondActivity,然后分别在两个Activity定义两个按钮,HomeActivity中的按钮用来启动SecondActivity,而SecondActivity中的按钮用来启动自己。并且在SecondActivity打印对应的生命周期方法,代码如下

public class HomeActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_home);
    }
    public void click(View view) {
        Intent intent = new Intent(this,SecondActivity.class);
        startActivity(intent);
    }
}
ublic class SecondActivity extends Activity {
    private static final String TAG = "SecondActivity";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        Log.d(TAG, "onCreate: ");
    }

    @Override
    protected void onStart() {
        super.onStart();
        Log.d(TAG, "onStart: ");
    }

    @Override
    protected void onResume() {
        super.onResume();
        Log.d(TAG, "onResume: ");
    }

    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        Log.d(TAG, "onNewIntent: ");
    }

    public void click(View view){
        Intent intent = new Intent(this,SecondActivity.class);
        startActivity(intent);
    }
}

首先我们不指定SecondActivity的启动模式,也就是默认的正常启动,我们在HomeActivity中启动它,然后自己又启动自己
这里写图片描述
可以看到,每次启动SecondActivity,都会重新创建实例,走新的生命周期方法。

实验二

我们把SecondActivity的启动模式设置为singleTop,首先解释一下singleTop的含义,如果当前栈顶存在该Activity,则此Activity不会被重建,启动该Activity时会调用onNewIntent方法,否则会重建该Activity,然后放到栈顶。
这里写图片描述

可以看到,当我们设置SecondActivity的启动模式为singleTop的时候,因为此时它在栈顶,我们再次去启动它,不会走onCreate和onStart,只会调用onNewIntent和onResume方法。

实验三

我们把SecondActivity的启动模式设置为singleTask,然后在SecondActivity中启动一个新的ThirdActivity,最后在ThirdActivity又启动SecondActivity,我们看看在这种情况下的执行。

这里写图片描述

最后我们把SecondActivity的启动模式设置为singleInstance,经过测试,结果和singleTask一样。

总结

前提:ActivityA已经启动过,处于当前应用的Activity堆栈中;

当ActivityA的LaunchMode为SingleTop时,如果ActivityA在栈顶,且现在要再启动ActivityA,这时会调用onNewIntent()方法

当ActivityA的LaunchMode为SingleInstance,SingleTask时,如果已经ActivityA已经在堆栈中,那么此时会调用onNewIntent()方法

当ActivityA的LaunchMode为Standard时,由于每次启动ActivityA都是启动新的实例,和原来启动的没关系,所以不会调用原来ActivityA的onNewIntent方法

启动模式的应用场景

接下来我们总结一下启动模式的应用场景:

SingleTask:一般在我们应用的主页(HomeActivity)中指定这个启动模式,假如我们的应用跳到了其他页面,在返回主页面,如果不指定启动模式,在返回的过程中就会产生多个实例,

SingleTop:适合接收通知启动的内容显示页面。例如,某个新闻客户端的新闻内容页面,如果收到10个新闻推送,每次都打开一个新闻内容页面是很烦人的,还有登录页面,从多个地方可能跳到登录页面,如果不指定启动模式,每次都会创建新的实例。

在 Android 中,`Activity` 的 `onNewIntent(Intent intent)` 方法用于处理新的 `Intent`,通常在 `Activity` 已经存在且被重新使用时调用(例如使用 `singleTop` 或 `singleTask` 启动模式时)。然而,**Fragment 没有直接对应的 `onNewIntent()` 方法**。 不过,我们可以通过一些方式在 Fragment 中实现类似的功能。 --- ## ✅ Fragment 中如何响应新的 Intent? ### 方法一:通过宿主 Activity 传递 Intent 宿主 `Activity` 接收到新的 `Intent` 后,可以主动通知当前显示的 Fragment: ```kotlin class HostActivity : AppCompatActivity() { override fun onNewIntent(intent: Intent?) { super.onNewIntent(intent) setIntent(intent) // 更新当前 Intent val fragment = supportFragmentManager.findFragmentById(R.id.fragment_container) if (fragment is OnNewIntentListener) { fragment.onNewIntent(intent) } } interface OnNewIntentListener { fun onNewIntent(intent: Intent?) } } ``` Fragment 中实现接口: ```kotlin class MyFragment : Fragment(), HostActivity.OnNewIntentListener { override fun onNewIntent(intent: Intent?) { // 处理新的 Intent intent?.getStringExtra("key")?.let { value -> Log.d("MyFragment", "Received new intent with value: $value") } } } ``` --- ### 方法二:使用 `setArguments()` 或 `ViewModel` 更新数据 如果你只是想更新 Fragment 的数据,而不是完全响应新的 Intent,可以使用以下方式: #### 使用 `setArguments()` ```kotlin override fun onNewIntent(intent: Intent?) { super.onNewIntent(intent) val fragment = supportFragmentManager.findFragmentById(R.id.fragment_container) fragment?.arguments = intent?.extras } ``` Fragment 中在 `onCreate()` 或 `onCreateView` 中读取参数: ```kotlin override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) arguments?.getString("key")?.let { Log.d("MyFragment", "Arguments: $it") } } ``` #### 使用 `ViewModel` 推荐使用 `ViewModel` 或 `SavedStateHandle` 来共享数据,适合 MVVM 架构: ```kotlin class SharedViewModel : ViewModel() { private val _intentData = MutableLiveData<String>() val intentData: LiveData<String> = _intentData fun updateData(data: String) { _intentData.value = data } } ``` Activity 中更新: ```kotlin viewModel.updateData(intent.getStringExtra("key")) ``` Fragment 中观察: ```kotlin sharedViewModel.intentData.observe(viewLifecycleOwner, { data -> Log.d("MyFragment", "Received: $data") }) ``` --- ## ✅ 总结 | 方法 | 是否推荐 | 说明 | |------|----------|------| | 通过宿主 Activity 调用 Fragment 的自定义 `onNewIntent()` 方法 | ✅ 推荐 | 明确传递 Intent,适合需要处理 Intent 的场景 | | 使用 `setArguments()` | ✅ 可选 | 适合传递少量数据 | | 使用 `ViewModel` | ✅ 强烈推荐 | 更现代、解耦、适合 MVVM 架构 | --- ## ✅ 结论 **Fragment 没有直接的 `onNewIntent()` 方法**,但可以通过宿主 `Activity` 主动通知 Fragment,或使用 `ViewModel`、`arguments` 等方式间接实现类似功能。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值