Android 关闭多个视图Intent.FLAG_ACTIVITY_CLEAR_TOP用法

本文详细介绍了如何在Android应用中实现从任意活动(Activity)安全退出整个应用程序的方法。通过使用特定的Intent标志,如Intent.FLAG_ACTIVITY_CLEAR_TOP,并结合Activity的启动模式,可以确保应用内的所有Activity被正确地清理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

如果已经启动了四个Activity:A,B,C和D。在D Activity里,我们要跳到B Activity,同时希望C finish掉,可以在startActivity(intent)里的intent里添加flags标记,如下所示:

[java] view plaincopy
  1. Intent intent = new Intent(this, B.class);   
  2. intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);  
  3. startActivity(intent);  


这样启动B Activity,就会把D,C都finished掉,如果你的B Activity的启动模式是默认的(multiple) ,则B Activity会finished掉,再启动一个新的Activity B。  如果不想重新再创建一个新的B Activity,则在上面的代码里再加上:

[java] view plaincopy
  1. intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);  

 这样B Activity就会再创建一个新的了,而是会重用之前的B Activity,同时调用B Activity的onNewIntent()方法。

 

问题:
多activity中退出整个程序,例如从A->B->C->D,这时我需要从D直接退出程序。

网上资料:{
finish()和system(0)都只能退出单个activity。杀进程等的等方式都不行~~~
解决问题:
我们知道Android的窗口类提供了历史栈,我们可以通过stack的原理来巧妙的实现,这里我们在D窗口打开A窗口时在Intent中直接加入标志Intent.FLAG_ACTIVITY_CLEAR_TOP,再次开启A时将会清除该进程空间的所有Activity。
在D中使用下面的代码:

[java] view plaincopy
  1. Intent intent = new Intent();   
  2. intent.setClass(D.this, A.class);  
  3. intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); //注意本行的FLAG设置   
  4. startActivity(intent);  
  5. finish();  


关掉自己
在A中加入代码:

[java] view plaincopy
  1. Override  
  2. protected void onNewIntent(Intent intent) {  
  3. // TODO Auto-generated method stub  
  4. super.onNewIntent(intent);  
  5. //退出  
  6.  if ((Intent.FLAG_ACTIVITY_CLEAR_TOP & intent.getFlags()) != 0) {  
  7.  finish();  
  8.  }  
  9. }  


A的Manifest.xml配置成android:launchMode="singleTop"

原理总结:
一般A是程序的入口点,从D起一个A的activity,加入标识Intent.FLAG_ACTIVITY_CLEAR_TOP这个过程中会把栈中B,C,都清理掉。因为A是android:launchMode="singleTop"
不会调用oncreate(),而是响应onNewIntent()这时候判断Intent.FLAG_ACTIVITY_CLEAR_TOP,然后把A finish()掉。
栈中A,B,C,D全部被清理。所以整个程序退出了。


可以利用清理历史栈的方法,来巧妙关闭所有activity,首先用一个设置为不可见的activity A来启动程序,这个activity A的作用只是用来垫栈底,只有启动和退出程序才会用到这个activity,而你需要退出的时候,只需要跳转至这个activity A  ,并让A  finish自己就可以实现关闭所有的activity。

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. Intent intent = new Intent();  
  2. intent.setClass(B.this, A.class);     //B为你按退出按钮所在的activity  
  3. intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);  //最关键是这句  
  4. startActivity(intent);  

        Intent.FLAG_ACTIVITY_CLEAR_TOP使得处于栈底的A发挥推土机的作用,从最底层把栈里所有的activity都清理掉,再在自己的oncreate方法加一句finish结束自己,即可实现退出。不放心的话,可以在A的ondestroy方法中加上system.exit(0) ,连跳转过程中的线程也可以终止的。

       至于如何初始化这个activity A ,决定程序是启动还是退出,随便弄一个static类型布尔变量控制一下就可以了。(*^__^*)

package com.st.base.ui; import android.content.Intent; import android.graphics.Color; import android.os.Bundle; import android.view.View; import android.view.Window; import android.view.WindowManager; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.app.AppCompatActivity; import androidx.databinding.DataBindingUtil; import androidx.databinding.ViewDataBinding; import androidx.fragment.app.FragmentActivity; import androidx.lifecycle.ViewModel; import androidx.lifecycle.ViewModelProvider; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; public abstract class BaseActivity<V extends ViewDataBinding> extends AppCompatActivity implements IBaseView { protected V binding; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); //页面接受的参数方法 initParam(); //隐藏顶部和底部栏 transparentNavBar(getWindow()); //私有的初始化Databinding和ViewModel方法 initViewDataBinding(savedInstanceState); //初始化ui initView(); //私有的ViewModel与View的契约事件回调逻辑 registorUIChangeLiveDataCallBack(); //页面数据初始化方法 initData(); //页面事件监听的方法,一般用于ViewModel层转到View层的事件注册 initViewObservable(); //页面监听回调 initCallback(); } protected abstract void initView(); protected abstract void initCallback(); protected abstract void releaseCallback(); private void initViewDataBinding(Bundle savedInstanceState) { //DataBindingUtil类需要在project的build中配置 dataBinding {enabled true }, 同步后会自动关联android.databinding包 binding = DataBindingUtil.setContentView(this, initContentView(savedInstanceState)); //支持LiveData绑定xml,数据改变,UI自动会更新 binding.setLifecycleOwner(this); initViewDataModel(); } protected void initViewDataModel() { } /** * 初始化根布局 * * @return 布局layout的id */ public abstract int initContentView(Bundle savedInstanceState); protected int initVariableId() { return 0; } protected void registorUIChangeLiveDataCallBack() { } @Override public void initParam() { } @Override public ViewModel initViewModel() { return null; } @Override public void initData() { } @Override public void initViewObservable() { } /** * 跳转页面 * * @param clz 所跳转的目的Activity类 * @param bundle 跳转所携带的信息 */ public void startActivity(Class<?> clz, Bundle bundle) { Intent intent = new Intent(this, clz); if (bundle != null) { intent.putExtras(bundle); } startActivity(intent); } /** * 创建ViewModel * * @param cls * @param <T> * @return */ public <T extends ViewModel> T createViewModel(FragmentActivity activity, Class<T> cls) { return new ViewModelProvider(activity).get(cls); } @Override protected void onDestroy() { super.onDestroy(); if (binding != null) { binding.unbind(); } releaseCallback(); } protected void transparentNavBar(@NonNull final Window window) { window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS | WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION); window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_STABLE); window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS); //状态栏、导航栏透明 window.setStatusBarColor(Color.TRANSPARENT); window.setNavigationBarContrastEnforced(false); window.setNavigationBarColor(Color.TRANSPARENT); View decorView = window.getDecorView(); int vis = decorView.getSystemUiVisibility(); int option = View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_STABLE; decorView.setSystemUiVisibility(vis | option); } } --------- 这个代码的单元测试可以通过PowerMock和mockito实现吗
最新发布
07-25
代码报错: java.lang.UnsupportedOperationException: remove,代码如下:package com.example.myapplication; import android.content.Intent; import android.content.SharedPreferences; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.Button; import android.widget.TextView; import android.widget.Toast; import androidx.appcompat.app.AppCompatActivity; import java.util.Arrays; import java.util.List; public class DetailActivity extends AppCompatActivity { String flag=""; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_detail); TextView name=(TextView)findViewById(R.id.name); TextView phone=(TextView) findViewById(R.id.phone); Button common_used=(Button)findViewById(R.id.common_used); Button back=(Button)findViewById(R.id.back); Intent intent=getIntent(); String user_name="姓名:"+intent.getStringExtra("name"); String user_phone="手机号码:"+intent.getStringExtra("number"); String position=intent.getStringExtra("position"); name.setText(user_name); phone.setText(user_phone); common_used.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { //存入 SharedPreferences sharedPreferences=getSharedPreferences("people",MODE_PRIVATE); SharedPreferences.Editor editor=sharedPreferences.edit(); editor.putString("Isstar",""); String FirstPosition=sharedPreferences.getString("Isstar",""); List<String> result = Arrays.asList(FirstPosition.split(",")); if(result.contains(position)){//点击前位置是星 flag="unused"; while (result.contains(position)) { result.remove(position); } String newIsstar=String.join(",",result); editor.putString("Isstar",newIsstar); Toast.makeText(DetailActivity.this,"删除此重点联系人", Toast.LENGTH_SHORT).show(); Toast.makeText(DetailActivity.this,newIsstar, Toast.LENGTH_SHORT).show(); // Log.i("新号","星号有:"+newIsstar); } else{ editor.putString("Isstar",FirstPosition+","+position); Toast.makeText(DetailActivity.this,"添加为重点联系人", Toast.LENGTH_SHORT).show(); flag="used"; String ceshi=sharedPreferences.getString("Isstar",""); Toast.makeText(DetailActivity.this,ceshi, Toast.LENGTH_SHORT).show(); } editor.commit(); } }); back.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent_back=new Intent(DetailActivity.this,PhoneActivity.class); intent_back.putExtra("position",position); // Toast.makeText(DetailActivity.this,String.valueOf(position), Toast.LENGTH_SHORT).show(); intent_back.putExtra("flag",flag); // setResult(RESULT_OK,intent_back); // finish(); startActivity(intent_back); } }); } }
07-13
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值