码迷,mamicode.com
首页 > 其他好文 > 详细

Intent及其七大属性及intent-filter设置

时间:2016-05-23 15:07:40      阅读:235      评论:0      收藏:0      [点我收藏+]

标签:



Intent及其七大属性


一、任务与回退栈:
(一)、任务Task:
①. 概念:
一个任务(task)就是在执行某项工作时与用户进行交互的Activity的集合。这些Ac

tivity按照被打开的顺序依次被安排在一个堆栈中(回退栈)。
②. 主屏页面:
设备的主屏是大多数任务的启动位置,当用户触摸一个应用程序启动器图标(或者a

pp快捷图标),应用程序的任务就会在前台显示。如果相关应用程序的任务不存在\

,那么就会有一个新的任务被创建,并且应用程序打开的“主”Activity会作为任务

中的根Activity。

(二)、回退栈:
①. 概念:
在当前的Activity启动了另一个Activity时,这个新的Activity被放到了堆栈的顶部

,并且带有焦点。前一个Activity并没有消失,而是保存在回退栈中,此时它处于停

止状态
当用户按下回退按钮时,当前的Activity会被从回退栈的顶部弹出(这个Activity被

销毁)而前一个Activity被恢复。堆栈中的Activity不会被重新排列。因此,回退栈

的操作跟后

进先出的栈对象结构是一样的。
在用户按下回退按钮时,当前Activity被销毁,并且前一个Activity被恢复。如果用户继续

按回退按钮,那么回退栈中的每个Activity会被依次弹出,前一个Activity会被显示,直到

用户返回主屏(或者返回到任务开始时运行的那个Activity)。当所有的Activity从回退栈

中被删除时,这个任务就不再存在了。

图1:用一个时间表显示了当前回退堆栈中的Activity之间在每个时间点的处理过程
技术分享
     ②. 多个任务:
技术分享
图2. 两个任务:任务B在前台接受用户交互,而任务A则在后台等待被恢复。

【注意:】后台中可以同时拥有多个任务,但是如果用户同时运行了很多后台任务,系统

为了回收内存可能销毁一些后台的Activity,从而导致Activity的状态丢失。
        因为回退堆栈中的Activity不曾被重新排列,因此如果允许用户从多个Activity中启动

一个特殊的Activity,那么就会创建一个新的Activity实例,并且在堆栈的顶部弹出(而不

是把之前的Activity实例带到堆栈的顶端)。这样在你的应用程序中一个Activity就可能被

实例化多次(甚至来自不同任务)。

(三)、Activity和Task的默认行为的总结:

①. 当Activity A启动Activity B时,ActivityA被终止,但是系统保留了它的状态(如滚动

条的位置和录入表单的文本)。如果用户在Activity B中按回退按钮,Activity A会使用被

保存的状态来进行恢复。
②. 当用户通过按主页(Home)按钮离开一个任务时,当前的Activity会被终止,并且被放

入后台。系统会保留任务中每个Activity的状态。如果用户随后通过选择启动图标来恢复

这个任务,那么任务会来到前台,并且恢复了堆栈顶部的Activity。
③. 如果用户按下回退按钮,当前的Activity会从堆栈中被弹出并且被销毁。堆栈中的前

一个Activity会被恢复。Activity被销毁时,系统不会保留Activity的状态。
④. Activity能够被实例化多次,甚至来自其他任务。

五、Activity启动模式

        在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化

操作。在Android中Activity的启动模式决定了Activity的启动运行方式。Android中Activity的启动

模式分为四种:

(一)、Activity启动模式设置:
        
<activity android:name=".MainActivity" android:launchMode="standard" />

(二)、Activity的四种启动模式:

    ①. standard(备注:standard是系统默认的启动模式。)

        标准启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。每个窗体的

getTaskId()保持不变,但是this.hashCode()发生改变。

    ②. singleTop

        如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,而不会创建新的Activ

ity对象,不过它会调用onNewIntent()方法。如果栈顶部不存在就会创建新的实例并放入

栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。会回调onNewI

ntent()方法。

    ③. singleTask

        如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。

重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实

例,将会创建新的实例放入栈中。 

        和singleTop在名字上即可看出区别,即singleTop每次只检测当前栈顶的Activity是

否是我们需要请求创建的,而singleTask则会检测栈中全部的Activity对象,从上向下,

如果检测到是我们所请求的,则会消灭此Activity对象上面的对象,直接把检测到的我们需

要的Activity置为栈顶。

    ④. singleInstance

        与singleTask模式的区别是,在singleInstance模式中,存放activity实例的(窗口对象)的

回退栈不能有其他任何窗口对象。因此如果该窗口不存在,则要新建任务来存放该singleI

nstance模式窗口。也就是说getTaskId()会发现任务id发生了变化。

        此启动模式和我们使用的浏览器工作原理类似,在多个程序中访问浏览器时,如果当

前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。此模式会节省大

量的系统资源,因为他能保证要请求的Activity对象在当前的栈中只存在一个。总之,在

开发Android项目时,巧妙设置Activity的启动模式会节省系统开销和提高程序运行效率

Intent及其七大属性及intent-filter设置
 
一、知识点回顾:Activity
(一)、如何实现Activity页面跳转?
示例代码:
//第一种方式:
Intent intent = new Intent(MainActivity.this,NextActivity.class);
startActivity(intent);

//第二种方式:
Intent intent = new Intent();
intent.setClass(MainActivity.this, NextActivity.class);
startActivity(intent);

//其实还有很多种Intent实现页面跳转的写法。

二、Intent对象介绍:
(一)、Intent基本介绍:
1、Intent 用于封装程序的”调用意图“。两个Activity之间,可以把需要交换的数据封装成Bundle对象,然后使用Intent携带Bundle对象,实现两个Activity之间的数据交换
2、Intent还是各种应用程序组件之间通信的重要媒介。不管想启动一个AcitivityService还是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()
2、启动Service:【后面详细讲】
    • ComponetName   startService()
    • boolean    bindService()
3、启动BroadcastReceiver:【后面详细讲】
    • 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的图标会出现在哪些启动的上下文环境里。
(一)、ComponentName属性:
1、指定了ComponentName属性的Intent已经明确了它将要启动哪个组件,因此这种Intent被称为显式Intent没有指定ComponentName属性的Intent被称为隐式Intent。隐式Intent没有明确要启动哪个组件,应用会根据Intent指定的规则去启动符合条件的组件。

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);

//也就是说,平时我们;;;

(二)、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
http://www.baidu.com:8080/a.jpg

    2、系统内置的几个Data属性常量:
    •   tel://:号码数据格式,后跟电话号码。
    •   mailto://:邮件数据格式,后跟邮件收件人地址。
    •   smsto://:短息数据格式,后跟短信接收号码。
    •   content://:内容数据格式,后跟需要读取的内容。
    •   file://:文件数据格式,后跟文件路径。
    •   market://search?q=pname:pkgname:市场数据格式,在Google Market里搜索包名为pkgname的应用。
    •   geo://latitude, longitude:经纬数据格式,在地图上显示经纬度所指定的位置。

    3、Intent利用Action属性和Data属性启动Android系统内置组件的代码:【不需要记忆,用到的时候查找资料】
(1)、拨打电话:
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); 

(2)、利用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系统内置组件的代码:
(1)、调用发送短信的程序 
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);

(2)、发送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也会产生新的页面,而不是重用之前的页面。】


四、Activity的launchMode【知识点回顾】:
    <activity
            android:name="com.example.s37demo.MainActivity"
            android:launchMode=""
            android:label="@string/app_name" >

   1、standard: (备注:standard是系统默认的启动模式。)
        标准启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。
        
        如果启动此Activity的Intent中没有设置FLAG_ACTIVITY_NEW_TASK标志, 则这个Activity与启动他的Activity在同一个Task中,如果设置了Activity请参考上面FLAG_ACTIVITY_NEW_TASK的详细说明,"launchMode"设置为"standard"的Activity可以被实例化多次, 可以在Task中的任何位置, 对于一个新的Intent请求就会实例化一次.

   2、singleTop
        如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,而不会创建新的Activity对象,不过它会调用onNewIntent()方法。如果栈顶部不存在就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。

        如果启动此Activity的Intent中没有设置FLAG_ACTIVITY_NEW_TASK标志, 则这个Activity与启动他的Activity在同一个Task中,如果设置了Activity请参考上面FLAG_ACTIVITY_NEW_TASK的详细说明,"launchMode"设置为"singleTop"的Activity可以被实例化多次,  可以在Task中的任何位置, 对于一个新的Intent请求如果在Task栈顶, 则会用栈顶的Activity响影Intent请求,而不会重新实例化对象接收请求, 如果没有在栈顶, 则会实例化一个新的对象接收Intent请求.

   3、singleTask
        如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。 
        和singleTop在名字上即可看出区别,即singleTop每次只检测当前栈顶的Activity是否是我们需要请求创建的,而singleTask则会检测栈中全部的Activity对象,从上向下,如果检测到是我们所请求的则会消灭此Activity对象上面的对象,直接把检测到的我们需要的Activity置为栈顶。
        
        "launchMode"设置为"singleTask"的Activity总是在栈底, 只能被实例化一次, 它允许其它Activity压入"singleTask"的Activity所在的Task栈,如果有新的Intent请求有此标志的Activity, 则系统会清除有此标志的Task栈中的全部Activity,并把此Activity显示出来.

   4、singleInstance
        在一个新栈中创建该Activity实例,并让多个应用共享该Activity实例。一旦这种模式的Activity实例存在于某个栈中,任何应用再激活这个Activity时都会重用该栈中的实例,其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中。此启动模式和我们使用的浏览器工作原理类似,在多个程序中访问浏览器时,如果当前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。此模式会节省大量的系统资源,因为他能保证要请求的Activity对象在当前的栈中只存在一个。

         "launchMode"设置为"singleInstance"的Activity总是在栈底, 只能被实例化一次, 不允许其它的Activity压入"singleInstance"的Activity所在Task栈,  即整个Task栈中只能有这么一个Activity.

Intent及其七大属性


一、任务与回退栈:
(一)、任务Task:
①. 概念:
一个任务(task)就是在执行某项工作时与用户进行交互的Activity的集合。这些Ac

tivity按照被打开的顺序依次被安排在一个堆栈中(回退栈)。
②. 主屏页面:
设备的主屏是大多数任务的启动位置,当用户触摸一个应用程序启动器图标(或者a

pp快捷图标),应用程序的任务就会在前台显示。如果相关应用程序的任务不存在\

,那么就会有一个新的任务被创建,并且应用程序打开的“主”Activity会作为任务

中的根Activity。

(二)、回退栈:
①. 概念:
在当前的Activity启动了另一个Activity时,这个新的Activity被放到了堆栈的顶部

,并且带有焦点。前一个Activity并没有消失,而是保存在回退栈中,此时它处于停

止状态
当用户按下回退按钮时,当前的Activity会被从回退栈的顶部弹出(这个Activity被

销毁)而前一个Activity被恢复。堆栈中的Activity不会被重新排列。因此,回退栈

的操作跟后

进先出的栈对象结构是一样的。
在用户按下回退按钮时,当前Activity被销毁,并且前一个Activity被恢复。如果用户继续

按回退按钮,那么回退栈中的每个Activity会被依次弹出,前一个Activity会被显示,直到

用户返回主屏(或者返回到任务开始时运行的那个Activity)。当所有的Activity从回退栈

中被删除时,这个任务就不再存在了。

图1:用一个时间表显示了当前回退堆栈中的Activity之间在每个时间点的处理过程
技术分享
     ②. 多个任务:
技术分享
图2. 两个任务:任务B在前台接受用户交互,而任务A则在后台等待被恢复。

【注意:】后台中可以同时拥有多个任务,但是如果用户同时运行了很多后台任务,系统

为了回收内存可能销毁一些后台的Activity,从而导致Activity的状态丢失。
        因为回退堆栈中的Activity不曾被重新排列,因此如果允许用户从多个Activity中启动

一个特殊的Activity,那么就会创建一个新的Activity实例,并且在堆栈的顶部弹出(而不

是把之前的Activity实例带到堆栈的顶端)。这样在你的应用程序中一个Activity就可能被

实例化多次(甚至来自不同任务)。

(三)、Activity和Task的默认行为的总结:

①. 当Activity A启动Activity B时,ActivityA被终止,但是系统保留了它的状态(如滚动

条的位置和录入表单的文本)。如果用户在Activity B中按回退按钮,Activity A会使用被

保存的状态来进行恢复。
②. 当用户通过按主页(Home)按钮离开一个任务时,当前的Activity会被终止,并且被放

入后台。系统会保留任务中每个Activity的状态。如果用户随后通过选择启动图标来恢复

这个任务,那么任务会来到前台,并且恢复了堆栈顶部的Activity。
③. 如果用户按下回退按钮,当前的Activity会从堆栈中被弹出并且被销毁。堆栈中的前

一个Activity会被恢复。Activity被销毁时,系统不会保留Activity的状态。
④. Activity能够被实例化多次,甚至来自其他任务。

五、Activity启动模式

        在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化

操作。在Android中Activity的启动模式决定了Activity的启动运行方式。Android中Activity的启动

模式分为四种:

(一)、Activity启动模式设置:
        
<activity android:name=".MainActivity" android:launchMode="standard" />

(二)、Activity的四种启动模式:

    ①. standard(备注:standard是系统默认的启动模式。)

        标准启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。每个窗体的

getTaskId()保持不变,但是this.hashCode()发生改变。

    ②. singleTop

        如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,而不会创建新的Activ

ity对象,不过它会调用onNewIntent()方法。如果栈顶部不存在就会创建新的实例并放入

栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。会回调onNewI

ntent()方法。

    ③. singleTask

        如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。

重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实

例,将会创建新的实例放入栈中。 

        和singleTop在名字上即可看出区别,即singleTop每次只检测当前栈顶的Activity是

否是我们需要请求创建的,而singleTask则会检测栈中全部的Activity对象,从上向下,

如果检测到是我们所请求的,则会消灭此Activity对象上面的对象,直接把检测到的我们需

要的Activity置为栈顶。

    ④. singleInstance

        与singleTask模式的区别是,在singleInstance模式中,存放activity实例的(窗口对象)的

回退栈不能有其他任何窗口对象。因此如果该窗口不存在,则要新建任务来存放该singleI

nstance模式窗口。也就是说getTaskId()会发现任务id发生了变化。

        此启动模式和我们使用的浏览器工作原理类似,在多个程序中访问浏览器时,如果当

前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。此模式会节省大

量的系统资源,因为他能保证要请求的Activity对象在当前的栈中只存在一个。总之,在

开发Android项目时,巧妙设置Activity的启动模式会节省系统开销和提高程序运行效率

Intent及其七大属性及intent-filter设置
 
一、知识点回顾:Activity
(一)、如何实现Activity页面跳转?
示例代码:
//第一种方式:
Intent intent = new Intent(MainActivity.this,NextActivity.class);
startActivity(intent);

//第二种方式:
Intent intent = new Intent();
intent.setClass(MainActivity.this, NextActivity.class);
startActivity(intent);

//其实还有很多种Intent实现页面跳转的写法。

二、Intent对象介绍:
(一)、Intent基本介绍:
1、Intent 用于封装程序的”调用意图“。两个Activity之间,可以把需要交换的数据封装成Bundle对象,然后使用Intent携带Bundle对象,实现两个Activity之间的数据交换
2、Intent还是各种应用程序组件之间通信的重要媒介。不管想启动一个AcitivityService还是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()
2、启动Service:【后面详细讲】
    • ComponetName   startService()
    • boolean    bindService()
3、启动BroadcastReceiver:【后面详细讲】
    • 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的图标会出现在哪些启动的上下文环境里。
(一)、ComponentName属性:
1、指定了ComponentName属性的Intent已经明确了它将要启动哪个组件,因此这种Intent被称为显式Intent没有指定ComponentName属性的Intent被称为隐式Intent。隐式Intent没有明确要启动哪个组件,应用会根据Intent指定的规则去启动符合条件的组件。

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);

//也就是说,平时我们;;;

(二)、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
http://www.baidu.com:8080/a.jpg

    2、系统内置的几个Data属性常量:
    •   tel://:号码数据格式,后跟电话号码。
    •   mailto://:邮件数据格式,后跟邮件收件人地址。
    •   smsto://:短息数据格式,后跟短信接收号码。
    •   content://:内容数据格式,后跟需要读取的内容。
    •   file://:文件数据格式,后跟文件路径。
    •   market://search?q=pname:pkgname:市场数据格式,在Google Market里搜索包名为pkgname的应用。
    •   geo://latitude, longitude:经纬数据格式,在地图上显示经纬度所指定的位置。

    3、Intent利用Action属性和Data属性启动Android系统内置组件的代码:【不需要记忆,用到的时候查找资料】
(1)、拨打电话:
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); 

(2)、利用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系统内置组件的代码:
(1)、调用发送短信的程序 
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);

(2)、发送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也会产生新的页面,而不是重用之前的页面。】


四、Activity的launchMode【知识点回顾】:
    <activity
            android:name="com.example.s37demo.MainActivity"
            android:launchMode=""
            android:label="@string/app_name" >

   1、standard: (备注:standard是系统默认的启动模式。)
        标准启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。
        
        如果启动此Activity的Intent中没有设置FLAG_ACTIVITY_NEW_TASK标志, 则这个Activity与启动他的Activity在同一个Task中,如果设置了Activity请参考上面FLAG_ACTIVITY_NEW_TASK的详细说明,"launchMode"设置为"standard"的Activity可以被实例化多次, 可以在Task中的任何位置, 对于一个新的Intent请求就会实例化一次.

   2、singleTop
        如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,而不会创建新的Activity对象,不过它会调用onNewIntent()方法。如果栈顶部不存在就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。

        如果启动此Activity的Intent中没有设置FLAG_ACTIVITY_NEW_TASK标志, 则这个Activity与启动他的Activity在同一个Task中,如果设置了Activity请参考上面FLAG_ACTIVITY_NEW_TASK的详细说明,"launchMode"设置为"singleTop"的Activity可以被实例化多次,  可以在Task中的任何位置, 对于一个新的Intent请求如果在Task栈顶, 则会用栈顶的Activity响影Intent请求,而不会重新实例化对象接收请求, 如果没有在栈顶, 则会实例化一个新的对象接收Intent请求.

   3、singleTask
        如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。 
        和singleTop在名字上即可看出区别,即singleTop每次只检测当前栈顶的Activity是否是我们需要请求创建的,而singleTask则会检测栈中全部的Activity对象,从上向下,如果检测到是我们所请求的则会消灭此Activity对象上面的对象,直接把检测到的我们需要的Activity置为栈顶。
        
        "launchMode"设置为"singleTask"的Activity总是在栈底, 只能被实例化一次, 它允许其它Activity压入"singleTask"的Activity所在的Task栈,如果有新的Intent请求有此标志的Activity, 则系统会清除有此标志的Task栈中的全部Activity,并把此Activity显示出来.

   4、singleInstance
        在一个新栈中创建该Activity实例,并让多个应用共享该Activity实例。一旦这种模式的Activity实例存在于某个栈中,任何应用再激活这个Activity时都会重用该栈中的实例,其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中。此启动模式和我们使用的浏览器工作原理类似,在多个程序中访问浏览器时,如果当前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。此模式会节省大量的系统资源,因为他能保证要请求的Activity对象在当前的栈中只存在一个。

         "launchMode"设置为"singleInstance"的Activity总是在栈底, 只能被实例化一次, 不允许其它的Activity压入"singleInstance"的Activity所在Task栈,  即整个Task栈中只能有这么一个Activity.

Intent及其七大属性及intent-filter设置

标签:

原文地址:http://blog.csdn.net/wuqingyidongren/article/details/51480340

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!