- 浏览: 441839 次
- 性别:
- 来自: 成都
文章分类
最新评论
-
sunwang810812:
万分感谢中!!!!!这么多年终于看到一个可运行可解决的方案!! ...
POI 后台生成Excel,在前台显示进度 -
zzb7728317:
LZ正解
Spring Jackson AjaxFileUpload 没有执行回调函数的解决办法 -
sleeper_qp:
lz是在源码上修改的么? 源码的话你重新编译一遍了么? 可 ...
由nutch readseg -dump 中文编码乱码想到的…… -
shenjian430:
请问你改好的程序在写在哪了?
由nutch readseg -dump 中文编码乱码想到的…… -
yinxusen:
It seems to be the bug occur in ...
Mahout Local模式 执行example的注意点
转自:http://hi.baidu.com/xunlerei/blog/item/c8e8e51743b8f8164a90a703.html
为什么要写这篇文章?
网上有很多种退出方法,可实际上很多方法都不通用(在某个版本下可用,到了另一个版本就不行),或者方法的实际效果根本就和其描述不符(也不知道那些发帖的人测没测试过)。
但我们的需求又确实存在。在某些情况下,我们需要在应用中打开多个Activity,但如果仅仅使用finish()方法就不能在需要的时候达到一次性退出的效果,自己作为一个Android退出问题的受害者,通过良久思考和实际测试,找到了一个比较不错的,在2.1-2.2-2.3版本下都通用的完全退出方法(2.1版本也基本可以代表1.5~2.1版本)!
PS:测试全部在模拟器环境下进行
我首先进行一下说明,下面两种方法效果完全相同
1,android.os.Process.killProcess(android.os.Process.myPid()) ; (这是Dalvik VM的本地方法)
2,System.exit(0); (常规java、c#的标准退出法,返回值为0代表正常退出 )
之后我的说明全部以android.os.Process.killProcess(android.os.Process.myPid()) 方法为准。
另外,我后边所说的程序入口即为在AndroidManifest.xml中配置为如下语句的Activity
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
下面开始我们的Android完美退出之旅:
先从网上的一段说明说起:
A->B
B中执行android.os.Process.killProcess(android.os.Process.myPid());
结果是结束了B,然后重新启动A。
实际情况是这样:A为程序入口,B中调用killProcess(android.os.Process.myPid())操作,实际上是将程序入口A和执行该语句的Activity B都关闭,并重新启动新的程序入口A。
所以,如果过程是A->B->C
则实际情况是:A为程序入口,C中调用killProcess(android.os.Process.myPid())操作将程序入口A和执行该语句的Activity C都关闭,并重新启动新的程序入口A(在Activity窗口历史栈当中,旧A 被关闭,新A 仍然会被放置在 旧A 所在的栈位置,不会到达栈顶端)。
PS: 如果killProcess(android.os.Process.myPid())或System.exit(0)是在程序入口A处执行,则是将入口A关闭,不会再开启新的A.
有人要问了,B Activity呢? B还存在着,B Activity没有被关闭。
如何解决这个问题?
首先说明一点
android.os.Process.killProcess(android.os.Process.myPid()) ;语句执行之后,后边的代码都将不再执行;
而finish();或startActivity(A.this,B.class);语句在执行完成后仍旧会执行后续的代码。(使用Thread.sleep多次验证,不用担心finish()过后不能startActivity了,相反也一样)。
所以,我们就可以充分利用这一点,既然finish();和startActivity(A.this,B.class);语句在执行后仍然可以执行后续代码操作,那我们可以将之组合在一个代码片段中,即
startActivity(new Intent(B.this, C.class));
finish();
或
finish();
startActivity(new Intent(B.this, C.class));
都是可以的,我们在B中使用该代码段,既将B Activity关闭了,也打开了C Activity,之前的问题Done!
如果你还有D,E,F ... 那也一样,在每次跳转到下一个Activity时,将finish()一块用上。使用这种方式,多余的Activity就能够被关闭了。
最后 假设我们有下面一种需求,对上边的内容进行总结:
Activity的开启过程为 1.Index --> 2.A_Activity --> 3.B_Activity --> 4.Index,在4.Index中实现退出,Index为程序入口。
Index
退出:就是最简单的finish();
跳转:也是最简单的 startActivity(new Intent(Index.this, A_Activity.class));
A_Activity
退回到首界面:分两种情况
1,需要Index更新(我的Index就有这个需求,首页面色彩发生变化)
使用android.os.Process.killProcess(android.os.Process.myPid()) ;关闭自己和之前的Index,创建新的Index;
2,不需要新的Index,Index无变化
使用最简单的finish(),并且效率还要高些。
跳转:
finish();
startActivity(new Intent(A_Activity.this, C.class));
关闭自身,开启除 Index 之外的其它Activity。
B_Activity
操作与 A_Activity 相同。当然跳转语句变成了
finish();
startActivity(new Intent(B_Activity.this, Index.class));
*************************************************************************************************************
上边的内容就是这样了,下面我再告诉你另外一种方法,可以实现不关闭中途的Activity,而是在后边的操作中一次性关闭前边开启的所有Activity,可以满足一些人通过按返回键返回上一个界面的要求!
通过Android的窗口类提供的历史栈,巧妙利用stack的原理,我们在Intent中加入标志 Intent.FLAG_ACTIVITY_CLEAR_TOP。
Intent intent = new Intent();
intent.setClass(One.this, Two.class);
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
接下来在B窗口中需要退出时直接使用finish方法即可全部退出。
假定有如下需求:
1.Index --> 2.A_Activity --> 3.B_Activity
在3中设置 intent.setClass(B_Activity.this, Index.class);
跳转后,程序会从栈顶逐个向后查找,直到找到栈中最近的Index,然后将这一路找到的Activity全部关闭,包括1、2、3(也就不需要像我先前的方法一路finish了,也保留了途中经过的Activity),最后再自动建一个新的Index Activity放到栈顶的位置,接下来在Index窗口中使用finish方法即可退出。
如果没有理解,看这个例子:
如果3中设置的是 intent.setClass(B_Activity.this, A_Activity.class);
则是将2,3关闭,再新建一个4.A_Activity,栈中就变成了
1.Index --> 4.A_Activity,懂了吧!
值得注意的是,在下面的情况中
1.Index --> 2.A_Activity --> 3.B_Activity --> 4.Index --> 5.A_Activity
在5中使用intent.setClass(A_Activity.this, B_Activity.class);
结果不是
1.Index --> 2.A_Activity --> 6.B_Activity
而是
1.Index --> 2.A_Activity --> 3.B_Activity --> 4.Index --> 5.A_Activity --> 6.B_Activity,因为4(程序入口)的存在,所以5对栈的操作不会到达3,而是发现4、5中都没有B_Activity后,没有关闭任何Activity,只在栈顶端新建了一个6.B_Activity。
这也间接说明了Dalvik虚拟机的遍历算法只进行到最近的程序入口,就认为后边没有该程序的Activity了。所以良好的Android编程习惯是,新建一个程序入口时,一定把老程序入口关掉,这也解释了为什么用killProcess方法更新后的程序入口Index一定还是在栈中的老位置,而不是到栈顶端。
自定义一个Actiivty 栈,道理同上,不过利用一个单例模式的Activity栈来管理所有Activity。并提供退出所有Activity的方法。代码如下:
public class ScreenManager {
private static Stack<Activity> activityStack;
private static ScreenManager instance;
private ScreenManager(){
}
public static ScreenManager getScreenManager(){
if(instance==null){
instance=new ScreenManager();
}
return instance;
}
//退出栈顶Activity
public void popActivity(Activity activity){
if(activity!=null){
activity.finish();
activityStack.remove(activity);
activity=null;
}
}
//获得当前栈顶Activity
public Activity currentActivity(){
Activity activity=activityStack.lastElement();
return activity;
}
//将当前Activity推入栈中
public void pushActivity(Activity activity){
if(activityStack==null){
activityStack=new Stack<Activity>();
}
activityStack.add(activity);
}
//退出栈中所有Activity
public void popAllActivityExceptOne(Class cls){
while(true){
Activity activity=currentActivity();
if(activity==null){
break;
}
if(activity.getClass().equals(cls) ){
break;
}
popActivity(activity);
}
}
}
这个方法我挺喜欢。
**********************************************************************************************
还有一种比较流行的Android经典完美退出方法,使用单例模式创建一个Activity管理对象,该对象中有一个Activity容器(具体实现自己处理,使用LinkedList等)专门负责存储新开启的每一个Activity,并且容易理解、易于操作,非常不错!
MyApplication类(储存每一个Activity,并实现关闭所有Activity的操作)
public class MyApplication extends Application {
private List<Activity> activityList = new LinkedList<Activity>();
private static MyApplication instance;
private MyApplication()
{
}
//单例模式中获取唯一的MyApplication实例
public static MyApplication getInstance()
{
if(null == instance)
{
instance = new MyApplication();
}
return instance;
}
//添加Activity到容器中
public void addActivity(Activity activity)
{
activityList.add(activity);
}
//遍历所有Activity并finish
public void exit()
{
for(Activity activity:activityList)
{
activity.finish();
}
System.exit(0);
}
}
在每一个Activity中的onCreate方法里添加该Activity到MyApplication对象实例容器中
MyApplication.getInstance().addActivity(this);
在需要结束所有Activity的时候调用exit方法
MyApplication.getInstance().exit();
有人反映,如果程序崩溃,可能会导致该类的被迫关闭并重建,使得前期放入的Activity无法正常关闭。
—————————————分割线—————————————————
我对其他一些退出方法进行的一点介绍和点评(不到之处还请指正):
@restartPackage(getPackageName())(具体就不介绍了)
我在SDK2.1版本下开发的一款小软件,放到Android2.2或2.3操作系统上无法退出,因为restartPackage方法在SDK2.1以后版本已经被废弃不用了,理由是因为它不安全,可能关闭同其他应用程序共享的Service,而这个Service别人还要用呢,你给别人关了就不对了。
@有人说的终极退出方法:
Intent startMain = new Intent(Intent.ACTION_MAIN);
startMain.addCategory(Intent.CATEGORY_HOME);
startMain.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(startMain);
System.exit(0);
实际上这种方法只是返回了Home页面,如果你再次进入应用,你会发现进入的首界面是你先前没有关闭的Activity。
@调用系统隐藏forceStopPackage方法,这里是通过映射调用(也有其他方法)
Method method = null;
ActivityManager manager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
try {
method = Class.forName("android.app.ActivityManager").getMethod("forceStopPackage", String.class);
method.invoke(manager,getPackageName());
} catch (Exception e) {
Log.d("force",e.getMessage());
}
我在SDK2.2和2.3的测试结果是,出现NULLPointerException,弹出错误窗口,程序被迫关闭,和预想的正常退出有差别。不过我们可以通过修改基类实现自己的Thread.UncaughtExceptionHandler接口的uncaughtException方法,这样就不会有错误窗口弹出。程序完全退出。
@和上面一样,不过这是故意制造异常退出(上边是无意制造的异常),但我认为这毕竟是下策。
发表评论
-
【转】区分Activity的四种加载模式
2012-01-11 21:33 935转自http://marshal.easymorse.c ... -
【转】Android程序完全退出的三种方法
2012-01-11 14:38 1102以下为网络上找的文 ... -
在程序中,打开google手机自带的程序Maps
2012-01-11 14:32 955Uri uri = Uri.parse("ht ... -
android keystore
2012-01-11 11:55 1032C:\Program Files\Java\jdk1.6 ... -
解决ListView和ExpandableListView滚动时背景变黑
2012-01-10 11:24 1662今天做东西的时候发现ListView和Expandabl ... -
android中intent的作用
2012-01-10 00:25 25641 Intent.ACTION_MAIN St ... -
关于Android中Java与Javascript之间的传值研究
2012-01-10 00:23 3000Android中提供的WebView的功能不可谓不强大, ... -
Android自适应屏幕大小和layout布局(横屏|竖屏)
2011-12-27 14:27 2603一:不同的layout Android手机屏幕大小不一,有48 ... -
从sbf刷机包里提取需要的apk文件
2011-11-18 14:07 25721、首先必然要有刷机包sbf文件 2、用MotoAndroi ... -
MB502 使用心得
2011-08-20 00:34 36261、解决MOTO ME502用不了CMWAP的办法 第 ... -
如何在Android中点击overlay弹出带尾巴的气泡
2011-07-10 12:37 60331.在res/layout目录下建立pop view ... -
Android中点中overlay弹出带尾巴的气泡的实现
2010-11-17 20:59 8331.在res/layout目录下建立pop view的xml文 ... -
(转)Android中LocationManager的简单使用,获取当前位置
2010-11-16 21:22 2281Android中LocationManager的提供了一系 ... -
创建指南针View的例子(2)
2010-11-12 15:25 15447. 为了查看指南针,修改main.xml资源,使用你的C ... -
创建指南针View的例子(1)
2010-11-12 15:20 1121在接下来的例子里,你将通过扩展View类创 ... -
Android Bitmap和Canvas学习笔记
2010-11-12 15:18 1141位图是我们开发中最常用的资源,毕竟一个漂亮的界面对用户 ... -
android sqlite3 adb使用
2010-10-04 23:10 15431.首先,emulator必须跑起来。可以通过eclipse或 ...
相关推荐
### MTK-Android软件培训知识点概述 ...以上内容详细介绍了 MTK Android 平台的软件开发涉及的关键步骤和技术要点, 包括环境搭建、编译过程、调试方法等方面, 对于初学者和开发者来说是非常宝贵的参考资料。
- **2.1 安装Android SDK与ADT plug-in** 详细介绍了如何下载和安装Android SDK以及Eclipse插件ADT。这些是开始Android应用开发的基础工具。 - **2.2 建立第一个Android项目~HelloAndroid!** 通过创建一个...
教授如何将开发的应用部署到真实的Android设备上进行测试的方法,包括连接设备、调试等步骤。 #### 第3章:用户人机界面 - **3.1 更改与显示文字标签** 展示如何在界面上动态更改文本内容,以及如何使用...
本节也介绍了通过集成测试模块验证SDK是否集成成功的方法。3.1. 数据上报策略统计服务SDK会先把数据记录缓存在本地,然后根据开发者设定的数据上报策略,触发上报流程。SDK中提供了一系列数据上报策略供开发者选择,...
- 通过 `Windows -> Android SDK Manager` 更新或安装 SDK 版本。 - 创建 AVD (Android Virtual Device),用于模拟设备环境。 ##### 2.3 验证 Android SDK 环境变量 1. **验证 platform-tools 环境变量**:在...
### Android创新实验报告知识点 #### 实验目的 本次实验旨在基于Android平台,使用Java语言及相关的Android开发技术,设计并实现一款界面友好、功能全面的贪吃蛇游戏。通过此项目,参与者能够深入理解Android应用...
- 此方法负责创建虚拟机实例,是ART虚拟机启动的关键步骤之一。 - **1.3** `AndroidRuntime`的`onVmCreated`: - 在虚拟机创建完成后被调用,用于执行一些特定于虚拟机的操作,如初始化类加载器等。 - **1.4** `...
##### 2.2 Android SDK介绍 Android SDK(Software Development Kit)是Android应用程序开发的基础工具包,包含了用于开发Android应用程序所需的各种工具和文档。SDK中的关键组件包括: - **Android模拟器**:允许...
以及本地数据存储等基础知识,而且通过“Hello, Android”项目深入探讨了外部通信、基于位置的服务、内置SQLite数据库、3D图形、多点触控、小部件、兼容性测试,以及如何向AndroidMarket发布应用程序等内容。...
- **准备文件**:下载Java JDK、Eclipse、Android SDK等开发工具; - **安装**:安装Java JDK、Eclipse等; - **应用**:启动Eclipse,创建并运行Android项目。 通过以上步骤,用户可以成功地在Ubuntu虚拟机中安装...
3. **Android SDK**:至少需要 Android SDK 2.0 或更新版本,此示例使用的是 2.2 版本。 4. **ADT 插件**:Android Developer Tools (ADT) 插件要求不低于 10.0 版本,这里使用的是 10.0 版本。 **APK 的编译与运行...
NativeApp 使用传统原生态Android SDK来实现的应用 WebApp 基于浏览器来实现的一种应用 HybridApp 一种可以下载的Native App,其用户界面的全部或者部分元素在嵌入式浏览器组件(WebView之类的)里面运行 优雅降级...