一、任务与回退栈:(一)、任务Task:1、概念:一个任务(task)就是在执行某项工作时与用户进行交互的Activity的集合。这些Activity按照被打开的顺序依次被安排在一个堆栈中(回退栈)。
2、主屏页面:设备的主屏是大多数任务的启动位置,当用户触摸一个应用程序启动器图标(或者app快捷图标),应用程序的任务就会在前台显示。如果相关应用程序的任务不存在,那么就会有一个新的任务被创建,并且应用程序打开的“主”Activity会作为任务中的根Activity。
(二)、回退栈:1、概念:在当前的Activity启动了另一个Activity时,这个新的Activity被放到了堆栈的顶部,并且带有焦点。前一个Activity并没有消失,而是保存在回退栈中,此时它处于停止状态。当用户按下回退按钮时,当前的Activity会被从回退栈的顶部弹出(这个Activity被销毁),而前一个Activity被恢复。堆栈中的Activity不会被重新排列。因此,回退栈的操作跟后进先出的对象结构是一样的。在用户按下回退按钮时,当前Activity被销毁,并且前一个Activity被恢复。如果用户继续按回退按钮,那么回退栈中的每个Activity会被依次弹出,前一个Activity会被显示,直到用户返回主屏(或者返回到任务开始时运行的那个Activity)。当所有的Activity从回退栈中被删除时,这个任务就不再存在了。
图1:用一个时间表显示了当前回退堆栈中的Activity之间在每个时间点的处理过程
2、多个任务:图2. 两个任务:任务B在前台接受用户交互,而任务A则在后台等待被恢复。
【注意:】后台中可以同时拥有多个任务,但是如果用户同时运行了很多后台任务,系统为了回收内存可能销毁一些后台的Activity,从而导致Activity的状态丢失。
因为回退堆栈中的Activity不曾被重新排列,因此如果允许用户从多个Activity中启动一个特殊的Activity,那么就会创建一个新的Activity实例,并且在堆栈的顶部弹出(而不是把之前的Activity实例带到堆栈的顶端)。这样在你的应用程序中一个Activity就可能被实例化多次(甚至来自不同任务)。
(三)、Activity和Task的默认行为的总结:
1、当Activity A启动Activity B时,ActivityA被终止,但是系统保留了它的状态(如滚动条的位置和录入表单的文本)。如果用户在Activity B中按回退按钮,Activity A会使用被保存的状态来进行恢复。
2、当用户通过按主页(Home)按钮离开一个任务时,当前的Activity会被终止,并且被放入后台。系统会保留任务中每个Activity的状态。如果用户随后通过选择启动图标来恢复这个任务,那么任务会来到前台,并且恢复了堆栈顶部的Activity。
3、如果用户按下回退按钮,当前的Activity会从堆栈中被弹出并且被销毁。堆栈中的前一个Activity会被恢复。Activity被销毁时,系统不会保留Activity的状态。
4、Activity能够被实例化多次,甚至来自其他任务。
二、Activity启动模式: 在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化操作。在Android中Activity的启动模式决定了Activity的启动运行方式。Android总Activity的启动模式分为四种:(一)、Activity启动模式设置: <activity android:name=".MainActivity" android:launchMode="standard" />
(二)、Activity的四种启动模式:
1. standard(备注:standard是系统默认的启动模式。)
标准启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。每个窗体的getTaskId()保持不变,但是this.hashCode()发生改变。
2. singleTop
如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,而不会创建新的Activity对象,不过它会调用onNewIntent()方法。如果栈顶部不存在就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。会回调onNewIntent()方法。
3. singleTask
如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。
和singleTop在名字上即可看出区别,即singleTop每次只检测当前栈顶的Activity是否是我们需要请求创建的,而singleTask则会检测栈中全部的Activity对象,从上向下,如果检测到是我们所请求的则会消灭此Activity对象上面的对象,直接把检测到的我们需要的Activity置为栈顶。
4. singleInstance
与singleTask模式的区别是存放singleInstance模式窗口对象的回退栈不能有其他任何窗口对象。因此如果该窗口不存在,则要新建任务来存放该singleInstance模式窗口。也就是说getTaskId()会发现任务id发生了变化。
此启动模式和我们使用的浏览器工作原理类似,在多个程序中访问浏览器时,如果当前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。此模式会节省大量的系统资源,因为他能保证要请求的Activity对象在当前的栈中只存在一个。
总之,在开发Android项目时,巧妙设置Activity的启动模式会节省系统开销和提高程序运行效率。三、Intent对象介绍:(一)、Intent基本介绍:1、Intent 用于封装程序的”调用意图“。两个Activity之间,可以把需要交换的数据封装成Bundle对象,然后使用Intent携带Bundle对象,实现两个Activity之间的数据交换;2、Intent还是各种应用程序组件之间通信的重要媒介。不管想启动一个Acitivity、Service还是BroadcastReceiver,Android均使用统一的Intent对象来封装这种”启动意图“。很明显使用Intent提供了一致的编程模型;3、Intent还有一个好处,如果应用程序只是想启动具有某种特征的组件,并不想和某个具体的组件耦合,则可以通过在intent-filter中配置相应的属性进行处理,与stucts2中的MVC框架思路类似。4、Intent对象大致包括7大属性:ComponentName、 Action 、 Category 、 Data 、Type、 Extra 、Flag。
(二)、Intent启动不同组件的方法:1、启动Activity:
- startActivity()
- startActivtyForResult()
- ComponetName startService()
- boolean bindService()
- sendBroadcast()
- sendOrderedBroadcast ()
- sendStickyBroadcast()
- sendStickyOrderedBroadcast()
四、Intent的七大属性:【重要】Intent对象大致包括7大属性:ComponentName、 Action 、 Category 、 Data 、Type、 Extra 、Flag。
- Action作为标识符,代表一个Intent,当一个Activity需要外部协助处理时,就会发出一个Intent,如果一个程序能完成相应功能,只要在intent-filter加上这个这个intent就可以了。
- Data保存需要传递的数据格式,比如:tel://
- Extras保存需要传递的额外数据。
- Category表示Intent的种类,从android上启动Activity有多种方式,比如 程序列表、桌面图标、点击Home激活的桌面等等,Category则用来标识这些Activity的图标会出现在哪些启动的上下文环境里。
2、示例代码:Intent intent = new Intent();ComponentName cName = new ComponentName(MainActivity.this,NextActivity.class);intent.setComponent(cName);startActivity(intent);
//实际上,以上的写法都被简化为以下写法:Intent intent = new Intent(MainActivity.this,NextActivity.class); startActivity(intent);
//也就是说,平时我们最常用的Intent页面跳转的写法就调用的是显式Intent。
(二)、Action、Category属性与intent-filter配置: 通常,Action、Category属性结合使用,定义这两个属性都是在配置文件的<intent-filter>节点中。Intent通过定义Action属性(其实就是一段自定义的字符串),这样就可以把Intent与具体的某个Activity分离,实现了解耦。否则,每次跳转,都要写成类似new Intent(MainActivity.this,NextActivity.class)这样的形式,也就是说必须将要跳转的目标Activity的名字写出来,这样的编码其实是“硬编码”,并没有实现松耦合。调用Intent对象的setAction()方法实现页面跳转虽然略微复杂(需要在AndroidManifest.xml文件中配置),但是实现了解耦。
1、示例代码:Intent intent = new Intent(); intent.setAction("com.steven.android06lifecycle.nextactivity"); startActivity(intent);
//在配置文件中注册Activity的时候需要声明:<activity android:name="com.steven.android06lifecycle.NextActivity"><intent-filter><action android:name="com.steven.android06lifecycle.nextactivity" /><category android:name="android.intent.category.DEFAULT" /> </intent-filter></activity>
//当某个页面是默认启动页面时,需要定义Action、Category的属性必须为以下字符串:【设置任务入口】 <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter>
2、常用Action属性常量: Intent对象不仅可以启动本应用内的程序组件,也可以启动Android系统的其他应用的组件,包括系统内置的程序组件(需要设置权限)。
- ACTION_MAIN:(android.intent.action.MAIN)Android程序入口。
-
- 每个Android应用必须且只能包含一个此类型的Action声明。【如果设置多个,则哪个在前,执行哪个。】
- ACTION_VIEW: (android.intent.action.VIEW) 显示指定数据。
- ACTION_EDIT: (android.intent.action.EDIT) 编辑指定数据。
- ACTION_DIAL: (android.intent.action.DIAL) 显示拨号面板。
- ACTION_CALL: (android.intent.action.CALL) 直接呼叫Data中所带的号码。
- ACTION_ANSWER: (android.intent.action.ANSWER) 接听来电。
- ACTION_SEND: (android.intent.action.SEND) 向其他人发送数据(例如:彩信/email)。
- ACTION_SENDTO: (android.intent.action.SENDTO) 向其他人发送短信。
- ACTION_SEARCH: (android.intent.action.SEARCH) 执行搜索。
- ACTION_GET_CONTENT: (android.intent.action.GET_CONTENT) 让用户选择数据,并返回所选数据。
(三)、Category 属性: Category属性为Action增加额外的附加类别信息。CATEGORY_LAUNCHER意味着在加载程序的时候Acticity出现在最上面,而CATEGORY_HOME表示页面跳转到HOME界面。1、实现页面跳转到HOME界面的代码:【记忆】Intent intent = new Intent(); intent.setAction(Intent.ACTION_MAIN); intent.addCategory(Intent.CATEGOTY_HOME); startActivity(intent);
2、常用Category属性常量:
- CATEGORY_DEFAULT: (android.intent.category.DEFAULT) Android系统中默认的执行方式,按照普通Activity的执行方式执行。
- CATEGORY_HOME: (android.intent.category.HOME) 设置该组件为Home Activity。
- CATEGORY_PREFERENCE: (android.intent.category.PREFERENCE) 设置该组件为Preference。
- CATEGORY_LAUNCHER: (android.intent.category.LAUNCHER) 设置该组件为在当前应用程序启动器中优先级最高的Activity,通常与入口ACTION_MAIN配合使用。
- CATEGORY_BROWSABLE: (android.intent.category.BROWSABLE) 设置该组件可以使用浏览器启动。
(四)、Data属性: 1、Data属性通常用于向Action属性提供操作的数据。Data属性的值是个Uri对象。
Uri的格式如下:scheme://host:port/path
2、系统内置的几个Data属性常量:
- tel://:号码数据格式,后跟电话号码。
- mailto://:邮件数据格式,后跟邮件收件人地址。
- smsto://:短息数据格式,后跟短信接收号码。
- content://:内容数据格式,后跟需要读取的内容。
- file:///:文件数据格式,后跟文件路径。
- market://search?q=pname:pkgname:市场数据格式,在Google Market里搜索包名为pkgname的应用。
- geo://latitude, longitude:经纬数据格式,在地图上显示经纬度所指定的位置。
3、Intent利用Action属性和Data属性启动Android系统内置组件的代码:【不需要记忆,用到的时候查找资料】(一)、拨打电话:Intent intent=new Intent(); intent.setAction(Intent.ACTION_CALL); //intent.setAction("android.intent.action.CALL"); //以下各项皆如此,都有两种写法。intent.setData(Uri.parse("tel:1320010001"));startActivity(intent);
//调用拨号面板:Intent intent=new Intent();intent.setAction(Intent.ACTION_DIAL); intent.setData(Uri.parse("tel:1320010001"));startActivity(intent);
//调用拨号面板:Intent intent=new Intent();intent.setAction(Intent.ACTION_VIEW); intent.setData(Uri.parse("tel:1320010001"));startActivity(intent);
(二)、利用Uri打开浏览器、打开地图等:Uri uri = Uri.parse("http://www.google.com"); //浏览器 Uri uri=Uri.parse("geo:39.899533,116.036476"); //打开地图定位 Intent intent = new Intent(); intent.setAction(Intent.ACTION_VIEW);intent.setData(uri);startActivity(intent);
(五)、Type属性: 1、Type属性用于指定Data所指定的Uri对应的MIME类型。MIME只要符合“abc/xyz”这样的字符串格式即可。
2、 Intent利用Action、Data和Type属性启动Android系统内置组件的代码:(三)、播放视频:Intent intent = new Intent(); Uri uri = Uri.parse("file:///sdcard/media.mp4"); intent.setAction(Intent.ACTION_VIEW);intent.setDataAndType(uri, "video/*"); startActivity(intent);
(六)、Extra属性: 1、通过intent.putExtra(键, 值)的形式在多个Activity之间进行数据交换。
2、系统内置的几个Extra常量:
- EXTRA_BCC:存放邮件密送人地址的字符串数组。
- EXTRA_CC:存放邮件抄送人地址的字符串数组。
- EXTRA_EMAIL:存放邮件地址的字符串数组。
- EXTRA_SUBJECT:存放邮件主题字符串。
- EXTRA_TEXT:存放邮件内容。
- EXTRA_KEY_EVENT:以KeyEvent对象方式存放触发Intent的按键。
- EXTRA_PHONE_NUMBER:存放调用ACTION_CALL时的电话号码。
3、 Intent利用Action、Data和Type、Extra属性启动Android系统内置组件的代码:(四)、调用发送短信的程序 Intent intent = new Intent();intent.setAction(Intent.ACTION_VIEW);intent.setType("vnd.android-dir/mms-sms"); intent.putExtra("sms_body", "信息内容..."); startActivity(intent);
//发送短信息 Uri uri = Uri.parse("smsto:13200100001"); Intent intent = new Intent(); intent.setAction(Intent. ACTION_SENDTO );intent.setData(uri);intent.putExtra("sms_body", "信息内容..."); startActivity( intent );
//发送彩信,设备会提示选择合适的程序发送 Uri uri = Uri.parse("content://media/external/images/media/23"); //设备中的资源(图像或其他资源) Intent intent = new Intent(); intent.setAction(Intent. ACTION_SEND );intent.setType("image/png"); intent.putExtra("sms_body", "内容"); intent.putExtra(Intent.EXTRA_STREAM, uri); startActivity(it);
(五)、发送Email:Intent intent=new Intent(); intent.setAction(Intent. ACTION_SEND );String[] tos={"android1@163.com"}; String[] ccs={"you@yahoo.com"}; intent.putExtra(Intent.EXTRA_EMAIL, tos); intent.putExtra(Intent.EXTRA_CC, ccs);intent.putExtra(Intent.EXTRA_TEXT, "The email body text"); intent.putExtra(Intent.EXTRA_SUBJECT, "The email subject text"); intent.setType("message/rfc822"); startActivity(Intent.createChooser(intent, "Choose Email Client"));
4、 Intent利用Action属性中的ACTION_GET_CONTENT获取返回值://选择图片 requestCode 返回的标识Intent intent = new Intent(); intent.setAction(Intent. ACTION_GET_CONTENT );intent.setType( "image/*" ); Intent wrapperIntent = Intent.createChooser(intent, null);startActivityForResult(wrapperIntent, requestCode);
//添加音频Intent intent = new Intent();intent.setAction(Intent. ACTION_GET_CONTENT );intent.setType( "video/*" ); Intent wrapperIntent = Intent.createChooser(intent, null);startActivityForResult(wrapperIntent, requestCode);
//视频Intent intent = new Intent();intent.setAction(Intent. ACTION_GET_CONTENT );intent.setType( "video/*" ); Intent wrapperIntent = Intent.createChooser(intent, null);startActivityForResult(wrapperIntent, requestCode);
//录音Intent intent = new Intent();intent.setAction(Intent. ACTION_GET_CONTENT );intent.setType( "audio/amr" ); intent.setClassName("com.android.soundrecorder","com.android.soundrecorder.SoundRecorder");startActivityForResult(intent, requestCode);
(七)、Flags属性:Intent可调用addFlags()方法来为Intent添加控制标记。【重要】
1、FLAG_ACTIVITY_CLEAR_TOP:(效果同Activity LaunchMode的singleTask) 如果在栈中已经有该Activity的实例,就重用该实例。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。
2、FLAG_ACTIVITY_SINGLE_TOP:(效果同Activity LaunchMode的singleTop) 如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,而不会创建新的Activity对象。
3、FLAG_ACTIVITY_NEW_TASK:
【备注:】以下几个为了解。
4、FLAG_ACTIVITY_MULTIPLE_TASK: 5、FLAG_ACTIVITY_BROUGHT_TO_FRONT: 6、FLAG_ACTIVITY_RESET_TASK_IF_NEEDED:
示例代码:
Intent intent = new Intent(this, MainActivity.class); //将Activity栈中处于MainActivity主页面之上的Activity都弹出。 intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); startActivity(intent);
【备注:】【重点,需要认真理解】例如:如果依次启动了四个Activity:A、B、C、D。在D Activity里,跳到B Activity,同时希望D 和 C 都finish掉,可以在startActivity(intent)里的intent里添加flags标记,如下所示:
Intent intent = new Intent(this, B.class); intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); startActivity(intent);
这样启动B Activity的同时,就会把D、C都finished掉。如果B Activity的launchMode是默认的“standard”,则B Activity会首先finished掉旧的B页面,再启动一个新的Activity B。 如果不想重新再创建一个新的B Activity,而是重用之前的B Activity,可以将B Activity的launchMode设置为“singleTask”。【特别需要注意的是:在部分手机中,如三星手机。即便是singleTask也会产生新的页面,而不是重用之前的页面。】