四大组件枢纽---Intent (意图)

四大组件间的 枢纽——Intent(意图),Android通信的桥梁

Intent (意图)

哪里用到了?

1、startActivity(Intent)/startActivityForResult(Intent):来启动一个Activity

2、startService(Intent)/bindService(Intent):来启动一个Service

3、sendBroadcast:发送广播到指定BroadcastReceiver

4、另外我们在注册四大组件时,写得很多的Intent-Filter

1、显式Intent与隐式Intent

  • 显式Intent:通过组件名指定启动的目标组件,比如startActivity(new Intent(A.this,B.class)); 每次 启动的组件只有一个
  • 隐式Intent:不指定组件名,而指定Intent的Action,Data,或Category,当我们启动组件时, 会去匹配 AndroidManifest.xml相关组件的Intent-filter,逐一匹配出满足属性的组件,当不止一个满足时, 会 弹出一个让我们选择启动哪个的对话框

Intent 中包含的主要信息如下:

1. ComponentName: 组件名称

指定Intent的目标组件的类名称。组件名称是可选的,如果填写,Intent对象会发送给指定组 件名称的组件,否则也可以通过其他Intent信息定位到适合的组件。组件名称是个 ComponentName类型的对象

Intent intent = new Intent();
ComponentName cn = new ComponentName(IntentActivity.this,OtherActivity.class);
in.setComponent(cn);
startActivity(intent);

2. action : 该activity可以执行的动作

该标识用来说明这个activity可以执行哪些动作,所以当隐式intent传递过来action时,如果跟这里所列出的任意一个匹配的话,就说明这个activity是可以完成这个intent的意图的,可 以将它激活!

常用的Action如下所示:

ACTION_CALL activity 启动一个电话.
ACTION_EDIT activity 显示用户编辑的数据.
ACTION_MAIN activity 作为Task中第一个Activity启动
ACTION_SYNC activity 同步手机与数据服务器上的数据.
ACTION_BATTERY_LOW broadcast receiver 电池电量过低警告.
ACTION_HEADSET_PLUG broadcast receiver 插拔耳机警告
ACTION_SCREEN_ON broadcast receiver 屏幕变亮警告.
ACTION_TIMEZONE_CHANGED broadcast receiver 改变时区警告.

两条原则:

  • 一条元素至少应该包含一个,否则任何Intent请求都不能和该 匹配。
  • 如果Intent请求的Action和中个任意一条匹配,那么该Intent 就可以激活该activity(前提是除了action的其它项也要通过)。 

两条注意:

  • 如果Intent请求或中没有说明具体的Action类型,那么会出现下面两种情 况。如果中没有包含任何Action类型,那么无论什么Intent请求都无法和这 条匹配。
  •  反之,如果Intent请求中没有设定Action类型,那么只要 中包含有Action类型,这个Intent请求就将顺利地通过 的行为测试。 

3. data: 执行时要操作的数据

在目标标签中包含了以下几种子元素,他们定义了url的匹配规则:

  • android:scheme 匹配url中的前缀,除了“http”、“https”、“tel”...之外,我们可以定义自己的前缀
  • android:host 匹配url中的主机名部分,如“google.com”,如果定义为“*”则表示任意主机名
  • android:port 匹配url中的端口 android:path 匹配url中的路径
<activity android:name=".TargetActivity">
<intent-filter>
<action android:name="com.scott.intent.action.TARGET"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:scheme="scott"
android:host="com.scott.intent.data"
android:port="7788"
android:path="/target"/>
</intent-filter>
</activity>

注意: 这个标识比较特殊,它定义了执行此Activity时所需要的数据,也就是说,这些数据是必须 的!!!!!所有如果其它条件都足以激活该Activity,但intent却没有传进来指定类型的Data 时,就不能激活该activity!!!!  

4. Category: 指定当前动作(Action)被执行的环境

即这个activity在哪个环境中才能被激活。不属于这个环境的,不能被激活

常用的Category属性如下所示:

CATEGORY_DEFAULT:Android系统中默认的执行方式,按照普通Activity的执行方式执行。表示所有intent都可以激活它
CATEGORY_HOME:设置该组件为Home Activity。
CATEGORY_PREFERENCE:设置该组件为Preference。
CATEGORY_LAUNCHER:设置该组件为在当前应用程序启动器中优先级最高的Activity,通常为入口ACTION_MAIN配合使用。(可以在AndroidManifest.xml文件查看)
CATEGORY_BROWSABLE:设置该组件可以使用浏览器启动。表示该activity只能用来浏览网页。
CATEGORY_GADGET:设置该组件可以内嵌到另外的Activity中。 

注意:如果该activity想要通过隐式intent方式激活,那么不能没有任何category设置,至少包含 一个android.intent.category.DEFAULT

5. Extra

Extra Extras属性主要用于传递目标组件所需要的额外的数据。通过putExtras()方法设置

 btn2.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent in = new Intent(FirstActivity.this, ThirdActivity.class);
                //1.传单个数据
//                in.putExtra("test","TTIT");
//                in.putExtra("number",100);
                //2.传多个数据
                Bundle b = new Bundle();
                b.putInt("number", 100);
                b.putString("test", "TTIT");
                in.putExtras(b);
//                startActivity(in);
                //返回数据给FirstActivity
//                startActivityForResult(in, 1001);
            }
        });

常用值如下所示:

EXTRA_CC:存放邮件抄送人地址的字符串数组。
EXTRA_EMAIL:存放邮件地址的字符串数组。
EXTRA_SUBJECT:存放邮件主题字符串。
EXTRA_TEXT:存放邮件内容。
EXTRA_KEY_EVENT:以KeyEvent对象方式存放触发Intent的按键。
EXTRA_PHONE_NUMBER:存放调用ACTION_CALL时的电话号码

6. IntentFlag (待补充)

在 Intent 类中定义的、充当 Intent 元数据的标志。标志可以指示 Android 系统如何启动 Activity(例如,Activity 应属于哪个任务),以及启动之后如何处理(例如,它是否属于最 近的 Activity 列表)。

界面跳转和服务的启动都会用到Intent,现在介绍Intent Flag是关于Activity的跳转
  Intent intent = new Intent(this,xxx.class);
  //1、如果activity在task存在,拿到最顶端,不会启动新的Activity
  intent.addFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT);
  //2、如果activity在task存在,将Activity之上的所有Activity结束掉
  intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  //3、默认的跳转类型,将Activity放到一个新的Task中
  intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  //4、如果Activity已经运行到了Task,再次跳转不会在运行这个Activity
  intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
  Intent Flag有20多种,我用过的就只有4中,后期用到了,知道什么作用,再向大家告知

2、Action、Category、Data以及匹配规则

  • Action:定义匹配动作,属性值为一个字符串
  • Category:属性用于指定当前动作(Action)被执行的环境。
  • Data:用于指定数据,通常是URI格式

Intent的Action、Category属性都是一个普通的字符串,其中Action代表该Intent所要完成的一个抽象“动作“,而Category则用于为Action增加额外的附加类别信息。通常Action属性会与Category属性结合使用。

Action要完成的只是一个抽象的动作,这个动作具体由哪个组件(或许是Activity,或者是BroadcastReceiver)来完成,Action这个字符串本身并不管。比如Android提供的标准Action:Intent.ACTION _VIEW,它只表示一个抽象的查看操作,但具体查看什么、启动哪个Activity来查看,Intent. ACTION _VIEW并不知道。这取决于Activity的< intent-filter>配置,只要某个Activity的< intent-filter>配置中包含了该ACTION _VIEW,该Activity就有可能被启动。

要注意的是:

  • 一个Intent对象最多只能包含一个Action属性,程序可调用Intent的setAction(String str)方法来设置Action属性值
  • 一个Intent对象可以包含多个Category属性,程序可调用Intent的addCategory(String str)方法来为Category属性
  • 当程序创建Intent时,该Intent默认启动Category属性值为Intent.CATEGORY_DEFAULT常量(常量值为android.intent.category.DEFAULT)的组件

1. Action匹配只要有一个与Intent中携带Action相同即可:

<activity android:name=".androidImplicitFilter.ActionActivity">
        <intent-filter>
                <action android:name=“com.action.a” />
                <action android:name=“com.action.b” />
                <action android:name=“com.action.c” />
                <category android:name=“android.intent.category.DEFAULT” />
        </intent-filter>
</activity>

Intent中只要携带"com.action.a"、“com.action.b”、"com.action.c"中的一个就可以启动ActionActivity

2. Category匹配要包含全部Intent中携带Category:

Category是一个字符串,系统中预定义了一些Category,也可以自定义Category

Intent intent = new Intent(“com.action.a”);
intent.addCategory(“com.category.a”);
intent.addCategory(“com.category.b”);
startActivity(intent);

intent包含2个Category ,"com.category.a"、"com.category.b"。要启动的组件必须包含这两个Category

<activity android:name=".androidImplicitFilter.ActionActivity">
         <intent-filter>
                <action android:name=“com.action.a” />
                <action android:name=“com.action.b” />
                <action android:name=“com.action.c” />
                <category android:name=“android.intent.category.DEFAULT” />
                <category android:name=“com.category.a” />
                <category android:name=“com.category.b” />
                <category android:name=“com.category.c” />
         </intent-filter>
</activity>

注意:Intent默认会携带"android.intent.category.DEFAULT",所以隐式启动的组件必须包含 <category android:name=“android.intent.category.DEFAULT” />

 <activity
            android:name="com.ttit.core.activity.SecondActivity"
            android:theme="@style/Theme.AppCompat.Dialog">
            <intent-filter>
                <action android:name="com.ttit.core.action.SecondActivity" />
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>

3. Data匹配与Action相同

只要包含所需要的data即可
Data匹配规则与Action类似,Data一般由两部分组成mimeType和URI。 mineType指资源类型包括文本、图片、音视频等等

URI结构如下:scheme://host:port/(path|pathPrefix|pathPatten)

例如 http://www.myHost:8080/user/index.html  content://com.download/file/10089

假如我们要通过响应一个超连接打开相应的组件,我们可以这样实现:

<intent-filter>
         <action android:name=“android.intent.action.VIEW” />
         <category android:name=“android.intent.category.DEFAULT” />
         <category android:name=“android.intent.category.BROWSABLE” />
         <data
                android:host=“myHost.app”
                android:scheme=“myScheme” />
</intent-filter>

以上代码解释:

1、<action android:name=“android.intent.action.VIEW” /> 系统根据不同的Data类型,通过筛选出已注册的Application来显示数据

2、<category android:name=“android.intent.category.DEFAULT” />默认的Category ,隐式启动的组件必须包含

3、<category android:name=“android.intent.category.BROWSABLE” />设置该组件可以使用浏览器启动

4、Data响应协议为myScheme域名为myHost.app,即“myScheme://myHost.app”,还可以设置port、path等信息,更精确的过滤

这样既可通过点击超链接打开组件,注意微信等一些其他app对浏览器做了修改,会屏蔽这些事件

<activity
            android:name=“XXXActivity”
            android:screenOrientation=“portrait”
            android:theme="@style/myTheme"
            android:windowSoftInputMode=“adjustResize|stateHidden”>
            <intent-filter>
                <action android:name=“android.intent.action.VIEW” />
                <action android:name=“android.intent.action.SEND” />
                <category android:name=“android.intent.category.DEFAULT” />
                <data android:mimeType=“application/pdf” />
                <data android:mimeType=“application/msword” />
                <data android:mimeType=“text/html” />
                <data android:mimeType=“application/vnd.openxmlformats-officedocument.wordprocessingml.document” />
            </intent-filter>

 </activity>

当前注册的Activity组件就是一个可以打开html、pdf、doc、docx文件的组件

<action android:name=“android.intent.action.VIEW” />
 <action android:name=“android.intent.action.SEND” />
 就是我们通常在微信或者QQ中遇到的使用第三方应用打开或者本地文件选择“发送”选项的场景,这里的data过滤使用了mimeType,当然也可以同时使用mimeType和URI

<intent-filter>
        <action android:name=“android.intent.action.VIEW” />
        <action android:name=“android.intent.action.SEND” />
        <category android:name=“android.intent.category.DEFAULT” />
        <data android:mimeType=“text/html”
                android:host=“myHost.app”
                android:scheme=“myScheme”
                android:port=8888
                android:path=“xxxxx”/>
</intent-filter>

3、常用系统Intent合集

//===============================================================
//1.拨打电话
// 给移动客服10086拨打电话
Uri uri = Uri.parse("tel:10086");
Intent intent = new Intent(Intent.ACTION_DIAL, uri);
startActivity(intent);
//===============================================================
//2.发送短信
// 给10086发送内容为“Hello”的短信
Uri uri = Uri.parse("smsto:10086");
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
intent.putExtra("sms_body", "Hello");
startActivity(intent);
//3.发送彩信(相当于发送带附件的短信)
Intent intent = new Intent(Intent.ACTION_SEND);
intent.putExtra("sms_body", "Hello");
Uri uri = Uri.parse("content://media/external/images/media/23");
intent.putExtra(Intent.EXTRA_STREAM, uri);
intent.setType("image/png");
startActivity(intent);
//===============================================================
//4.打开浏览器:
// 打开百度主页
Uri uri = Uri.parse("http://www.baidu.com");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
//===============================================================
//5.发送电子邮件:(阉割了Google服务的没戏!!!!)
// 给someone@domain.com发邮件
Uri uri = Uri.parse("mailto:someone@domain.com");
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
startActivity(intent);
// 给someone@domain.com发邮件发送内容为“Hello”的邮件
Intent intent = new Intent(Intent.ACTION_SEND);
intent.putExtra(Intent.EXTRA_EMAIL, "someone@domain.com");
intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
intent.putExtra(Intent.EXTRA_TEXT, "Hello");
intent.setType("text/plain");
startActivity(intent);
// 给多人发邮件
Intent intent=new Intent(Intent.ACTION_SEND);
String[] tos = {"1@abc.com", "2@abc.com"}; // 收件人
String[] ccs = {"3@abc.com", "4@abc.com"}; // 抄送
String[] bccs = {"5@abc.com", "6@abc.com"}; // 密送
intent.putExtra(Intent.EXTRA_EMAIL, tos);
intent.putExtra(Intent.EXTRA_CC, ccs);
intent.putExtra(Intent.EXTRA_BCC, bccs);
intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
intent.putExtra(Intent.EXTRA_TEXT, "Hello");
intent.setType("message/rfc822");
startActivity(intent);
//===============================================================
//6.显示地图:
// 打开Google地图中国北京位置(北纬39.9,东经116.3)
Uri uri = Uri.parse("geo:39.9,116.3");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
//===============================================================
//7.路径规划
// 路径规划:从北京某地(北纬39.9,东经116.3)到上海某地(北纬31.2,东经121.4)
Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=39.9
116.3&daddr=31.2 121.4");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
//===============================================================
//8.多媒体播放:
Intent intent = new Intent(Intent.ACTION_VIEW);
Uri uri = Uri.parse("file:///sdcard/foo.mp3");
intent.setDataAndType(uri, "audio/mp3");
startActivity(intent);
//获取SD卡下所有音频文件,然后播放第一首=-=
Uri uri =
Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI,
"1");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
//===============================================================
//9.打开摄像头拍照:
// 打开拍照程序
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(intent, 0);
// 取出照片数据
Bundle extras = intent.getExtras();
Bitmap bitmap = (Bitmap) extras.get("data");
//另一种:
//调用系统相机应用程序,并存储拍下来的照片
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
time = Calendar.getInstance().getTimeInMillis();
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new
File(Environment
.getExternalStorageDirectory().getAbsolutePath()+"/tucue", time +
".jpg")));
startActivityForResult(intent, ACTIVITY_GET_CAMERA_IMAGE);
//===============================================================
//10.获取并剪切图片
// 获取并剪切图片
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("image/*");
intent.putExtra("crop", "true"); // 开启剪切
intent.putExtra("aspectX", 1); // 剪切的宽高比为1:2
intent.putExtra("aspectY", 2);
intent.putExtra("outputX", 20); // 保存图片的宽和高
intent.putExtra("outputY", 40);
intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp")));
// 保存路径
intent.putExtra("outputFormat", "JPEG");// 返回格式
startActivityForResult(intent, 0);
// 剪切特定图片
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setClassName("com.android.camera",
"com.android.camera.CropImage");
intent.setData(Uri.fromFile(new File("/mnt/sdcard/temp")));
intent.putExtra("outputX", 1); // 剪切的宽高比为1:2
intent.putExtra("outputY", 2);
intent.putExtra("aspectX", 20); // 保存图片的宽和高
intent.putExtra("aspectY", 40);
intent.putExtra("scale", true);
intent.putExtra("noFaceDetection", true);
intent.putExtra("output", Uri.parse("file:///mnt/sdcard/temp"));
startActivityForResult(intent, 0);
//===============================================================
//11.打开Google Market
// 打开Google Market直接进入该程序的详细页面
Uri uri = Uri.parse("market://details?id=" + "com.demo.app");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
//===============================================================
//12.进入手机设置界面:
// 进入无线网络设置界面(其它可以举一反三)
Intent intent = new
Intent(android.provider.Settings.ACTION_WIRELESS_SETTINGS);
startActivityForResult(intent, 0);
//===============================================================
//13.安装apk:
Uri installUri = Uri.fromParts("package", "xxx", null);
returnIt = new Intent(Intent.ACTION_PACKAGE_ADDED, installUri);
//===============================================================
//14.卸载apk:
Uri uri = Uri.fromParts("package", strPackageName, null);
Intent it = new Intent(Intent.ACTION_DELETE, uri);
startActivity(it);
//===============================================================
//15.发送附件:
Intent it = new Intent(Intent.ACTION_SEND);
it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/eoe.mp3");
sendIntent.setType("audio/mp3");
startActivity(Intent.createChooser(it, "Choose Email Client"));
//===============================================================
//16.进入联系人页面:
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setData(People.CONTENT_URI);
startActivity(intent);
//===============================================================
//17.查看指定联系人:
Uri personUri = ContentUris.withAppendedId(People.CONTENT_URI,
info.id);//info.id联系人ID
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setData(personUri);
startActivity(intent);
//===============================================================
//18.调用系统编辑添加联系人(高版本SDK有效):
Intent it = newIntent(Intent.ACTION_INSERT_OR_EDIT);
it.setType("vnd.android.cursor.item/contact");
//it.setType(Contacts.CONTENT_ITEM_TYPE);
it.putExtra("name","myName");
it.putExtra(android.provider.Contacts.Intents.Insert.COMPANY,
"organization");
it.putExtra(android.provider.Contacts.Intents.Insert.EMAIL,"email");
it.putExtra(android.provider.Contacts.Intents.Insert.PHONE,"homePhone"
);
it.putExtra(android.provider.Contacts.Intents.Insert.SECONDARY_PHONE,"
mobilePhone");
it.putExtra(
android.provider.Contacts.Intents.Insert.TERTIARY_PHONE,"workPhone");
it.putExtra(android.provider.Contacts.Intents.Insert.JOB_TITLE,"title"
);
startActivity(it);
//===============================================================
//19.调用系统编辑添加联系人(全有效):
Intent intent = newIntent(Intent.ACTION_INSERT_OR_EDIT);
intent.setType(People.CONTENT_ITEM_TYPE);
intent.putExtra(Contacts.Intents.Insert.NAME, "My Name");
intent.putExtra(Contacts.Intents.Insert.PHONE, "+1234567890");
intent.putExtra(Contacts.Intents.Insert.PHONE_TYPE,Contacts.PhonesColu
mns.TYPE_MOBILE);
intent.putExtra(Contacts.Intents.Insert.EMAIL, "com@com.com");
intent.putExtra(Contacts.Intents.Insert.EMAIL_TYPE,
Contacts.ContactMethodsColumns.TYPE_WORK);
startActivity(intent);
//===============================================================
//20.打开另一程序
Intent i = new Intent();
ComponentName cn = new ComponentName("com.example.jay.test",
"com.example.jay.test.MainActivity");
i.setComponent(cn);
i.setAction("android.intent.action.MAIN");
startActivityForResult(i, RESULT_OK);
//===============================================================
//21.打开录音机
Intent mi = new Intent(Media.RECORD_SOUND_ACTION);
startActivity(mi);
//===============================================================
//22.从google搜索内容
Intent intent = new Intent();
intent.setAction(Intent.ACTION_WEB_SEARCH);
intent.putExtra(SearchManager.QUERY,"searchString")
startActivity(intent);

  

 

posted @ 2021-03-27 20:02  恋人星  阅读(382)  评论(0编辑  收藏  举报