一、隐式意图介绍
显式意图我们前面已经提到,形如:
Intent intent = new Intent();
intent.setClass(this,Other.class);//此句表示显式意图,因为明确设置激活对象为Other类
startActivity(intent);
顾名思义,隐式意图就是在不明确设置激活对象的前提下寻找最匹配的组件,举个例子,比如有5个人:
(1)A:170cm
(2)B:160cm
(3)C:180cm
(4)D:190cm
(5)E:200cm
如果是显示意图的话,如果我们要指明选择A的话会说:”我选择A.“,但是如果是隐式意图,则会说:”我要选择170cm的人“,虽然没有指明要选A,但会寻找条件最匹配的人。
在intent过滤器中类似于上面例子中的”身高“条件的匹配条件有:
(1)action
(2)category
(3)data:scheme、host、path、type
当在程序中设置了这些激活组件的条件,程序就会去寻找最匹配的组件,但是注意:只要有一点不匹配,则就是不匹配;
比如:
Intent intent = new Intent();
intent.setAction("a");//此句只是指定了Action
startActivity(intent);//寻找最匹配的组件激活,内部会调用intent.addCategory("android.intent.category.DEFAULT");
二、隐式Intent的核心代码
首先是在AndroidManifest.xml中为某个Activity设置意图过滤器:
- <activity>
- <intent-filter>
- <action android:name="...."/>
- <category android:name="...."/>
- <category android:name="android.intent.category.DEFAULT"/> <!--此句一般都要加 -->
- <data android:scheme="..." android:host="..." android:path="/..." android:type="..."/>
- </intent-filter>
- </activity>
以上设置是设置Activity本身的属性,接下来在程序中要设置的是我们要寻找时匹配的条件:
(1)Intent intent = new Intent();
(2)intent.setAction("....");
(3)intent.addCategory("....");
(4)intent.setData(Uri.parse("...."));//设置data的scheme、host、path条件
(5)intent.setDataAndType(Uri.parse(""),String type);//同时设置data的scheme、host、path、type条件
(6)startActivity(intent);//调用intent.addCategory("android.intent.category.DEFAULT");
三、代码举例
场景介绍:在MainActivity中有一个按钮,点击按钮后,会进行隐式Intent匹配,最后寻找到并激活OtherActivity.
情况1:
- <activity
- android:name=".OtherActivity"
- android:label="OtherActivity" >
- <intent-filter>
- <action android:name="com.xiazdong.action" />
- <category android:name="android.intent.category.DEFAULT" />
- <category android:name="com.xiazdong.category" />
- <data
- android:host="www.xiazdong.com"
- android:scheme="xiazdong"/>
- </intent-filter>
- </activity>
则代码为:
- Intent intent = new Intent();
- intent.setAction("com.xiazdong.action");
- intent.addCategory("com.xiazdong.category");
- intent.setData(Uri.parse("xiazdong://www.xiazdong.com/xia"));
- startActivity(intent); //此方法中调用intent.addCategory("android.intent.category.DEFAULT");
情况2:
在<data>中多了一个android:mimeType="text/*",此时不能使用intent.setData,而要使用intent.setDataAndType();
- <activity
- android:name=".OtherActivity"
- android:label="OtherActivity" >
- <intent-filter>
- <action android:name="com.xiazdong.action" />
- <category android:name="android.intent.category.DEFAULT" />
- <category android:name="com.xiazdong.category" />
- <data
- android:host="www.xiazdong.com"
- android:scheme="xiazdong" android:mimeType="text/*"/>
- </intent-filter>
- </activity>
代码为:
- Intent intent = new Intent();
- intent.setAction("com.xiazdong.action");
- intent.addCategory("com.xiazdong.category");
- intent.setDataAndType(Uri.parse("xiazdong://www.xiazdong.com/xia"), "text/plain"); //匹配了text/*
- startActivity(intent); //此方法中调用intent.addCategory("android.intent.category.DEFAULT");
通过设置Action字符串,表明自己的意图,即我想干嘛,需要由系统解析,找到能够处理这个Intent的Activity并启动。比如我想打电话,则可以设置Action为"android.intent.action.DIAL"字符串,表示打电话的意图,系统会找到能处理这个意图的Activity,例如调出拨号面板。
有几点需要注意:
1、
这个Activity其他应用程序也可以调用,只要使用这个Action字符串。这样应用程序之间交互就很容易了,例如手机QQ可以调用QQ空间,可以调用腾讯微博等。
因为如此,为了防止应用程序之间互相影响,一般命名方式是包名+Action名,例如这里命名"abcdefg"就很不合理了,就应该改成"com.example.app016.MyTest"。
2、
当然,你可以在自己的程序中调用其他程序的Action。
例如可以在自己的应用程序中调用拨号面板:
- Intent intent = new Intent(Intent.ACTION_DIAL);
- // 或者Intent intent = new Intent("android.intent.action.DIAL");
- // Intent.ACTION_DIAL是内置常量,值为"android.intent.action.DIAL"
- startActivity(intent);
3、一个Activity可以处理多种Action
只要你的应用程序够牛逼,一个Activity可以看网页,打电话,发短信,发邮件。。。当然可以。
Intent的Action只要是其中之一,就可以打开这个Activity。
- <activity
- android:name="com.example.app016.SecondActivity">
- <intent-filter>
- <!-- 可以处理下面三种Intent -->
- <action android:name="com.example.app016.SEND_EMAIL"/>
- <action android:name="com.example.app016.SEND_MESSAGE"/>
- <action android:name="com.example.app016.DAIL"/>
- <category android:name="android.intent.category.DEFAULT" />
- </intent-filter>
- </activity>
对于一个Action字符串,系统有可能会找到一个Activity能处理这个Action,也有可能找到多个Activity,也可能一个都找不到。
1、找到一个Activity
很简单,直接打开这个Activity。这个不需要解释。
2、找到多个Acyivity
系统会提示从多个activity中选择一个打开。
例如我们自己开发一个拨号面板应用程序,可以设置activity的<intent-filter>中Action name为"android.intent.action.DIAL",这样别的程序调用拨号器时,用户可以从Android自带的拨号器和我们自己开发的拨号器中选择。
- <activity
- android:name="com.example.app016.SecondActivity">
- <intent-filter>
- <action android:name="android.intent.action.DIAL"/>
- <category android:name="android.intent.category.DEFAULT" />
- </intent-filter>
- </activity>
这也就是当Android手机装上UC浏览器后,打开网页时会弹出选择Android自带浏览器还是UC浏览器,可能都会遇到过。
3、一个Activity都没找到
一个都没找到的话,程序就会出错,会抛出ActivityNotFoundException。比如随便写一个action字符串:
- Intent intent = new Intent("asasasas");
- startActivity(intent);
所以应该注意try catch异常。
- Intent intent = new Intent("asasasas");
- try
- {
- startActivity(intent);
- }
- catch(ActivityNotFoundException e)
- {
- Toast.makeText(this, "找不到对应的Activity", Toast.LENGTH_SHORT).show();
- }
- Intent intent = new Intent(Intent.ACTION_DIAL);
- if(intent.resolveActivity(getPackageManager()) == null)
- {
- // 设置控件不可用
- }
注意resolveActivity方法返回值就是显式Intent上面讲到的ComponentName对象,一般情况下也就是系统找到的那个Activity。但是如果有多个Activity可供选择的话,则返回的Component是com.android.internal.app.ResolverActivity,也就是用户选择Activity的那个界面对应的Activity,这里不再深究。
- Intent intent = new Intent(Intent.ACTION_DIAL);
- ComponentName componentName = intent.resolveActivity(getPackageManager());
- if(componentName != null)
- {
- String className = componentName.getClassName();
- Toast.makeText(this, className, Toast.LENGTH_SHORT).show();
- }
参考: Android入门:隐式Intent http://blog.csdn.net/xiazdong/article/details/7764865
Android理解:显式和隐式Intent http://blog.csdn.net/xiao__gui/article/details/11392987