intent-filter 之 data 「scheme, host, port, mimeType, path, pathPrefix, pathPattern」

一、定义
 

scheme, host, port, path, pathPrefix, pathPattern 是用来匹配 Intent 中的 Data Uri 的。具体规则如下:
 

  scheme://host:port/path or pathPrefix or pathPattern
 

这里需要注意的是这里的 scheme 不是 schema,也许你记得 xmlns:android="http://schemas.android.com/apk/res/android" 这段声明,你就会想起其中的 schema (至少我是这样想到了...- -!),但这里的 scheme 不是 schema。虽然在写 AndroidManifest.xml 的时候,有智能提示,但是希望大家还是能注意到。

上面那句最后的 “path or pathPrefix or pathPattern” 是指后面的 path 验证可以使用 data 属性中的 android:path、android:pathPrefix 或 pathPattern,你可以添加任意个 data 标签,由于是 “or” ,因此,只要其中任意一个 data 匹配,系统就会选择你的 Activity 启动,当然,如果别的 Activity 也有相同的 data 标签,系统就会给用户弹出一个 Chooser Dialog。


mimeType 也是是用来匹配 Intent 的。比如,当你使用 Intent.setType("text/plain") ,那么系统将会匹配到所有注册 android:mimeType="text/plain" 的 Activity,想获取更多有关 mimeType 的知识请参考:【转】备份:Android 常用 mimeType 表

这里需要十分注意的是 Intent.setType(), Intent.setData,Intent.setDataAndType() 这三个方法!
 

  • setType 调用后设置 mimeType,然后将 data 置为 null;
  • setData 调用后设置 data,然后将 mimeType 置为 null;
  • setDataAndType 调用后才会同时设置 data 与 mimeType。
     
另外需要注意的是,如果你在 data 标签,既设置了 mimeType 又设置了 scheme 之内的,那么你的 Intent 需要同时设置匹配的 data 与 mimeType 即调用 setDataAndType ,系统才能匹配到这个 Activity(即便你 mimeType 设置为 "*/*" 也是如此)。当然,如果你没有设置 mimeType,那么调用 setData 进行匹配,如果你设置了任何的 mimeType 将不会匹配到该 Activity。

二、区别

  这里主要说的区别是 pathpathPrefixpathPattern 之间的区别
 
  • path 用来匹配完整的路径,如:http://example.com/blog/abc.html,这里将 path 设置为 /blog/abc.html 才能够进行匹配;
  • pathPrefix 用来匹配路径的开头部分,拿上面的 Uri 来说,这里将 pathPrefix 设置为 /blog 就能进行匹配了;
  • pathPattern 用表达式来匹配整个路径,这里需要说下匹配符号与转义。
匹配符号:
  1. “*” 用来匹配0次或更多,如:“a*” 可以匹配“a”、“aa”、“aaa”...
  2. “.” 用来匹配任意字符,如:“.” 可以匹配“a”、“b”,“c”...
  3. 因此 “.*” 就是用来匹配任意字符0次或更多,如:“.*html” 可以匹配 “abchtml”、“chtml”,“html”,“sdf.html”...
转义:
因为当读取 Xml 的时候,“\” 是被当作转义字符的(当它被用作 pathPattern 转义之前),因此这里需要两次转义,读取 Xml 是一次,在 pathPattern 中使用又是一次。如:“*” 这个字符就应该写成 “\\*”,“\” 这个字符就应该写成 “\\\\”。


三、一些例子
1.

通过用手机的浏览器(内置,第三方都可)访问一个网页,实现点击一个链接启动自己的应用,并传递数据。

首先在Mainifest文件里面对要启动的Activity添加一个过滤器。

  1. <activity  
  2.             android:name="com.example.helloworld.MainActivity"  
  3.             android:label="@string/app_name" >  
  4.             <intent-filter>  
  5.                 <action android:name="android.intent.action.MAIN" />  
  6.   
  7.                 <category android:name="android.intent.category.LAUNCHER" />  
  8.             </intent-filter>  
  9.             <intent-filter>  
  10.                 <action android:name="android.intent.action.VIEW"/>     
  11.                 <category android:name="android.intent.category.DEFAULT"/>               
  12.                 <category android:name="android.intent.category.BROWSABLE"/>  
  13.                 <data android:scheme="znn"/>  
  14.             </intent-filter>  
  15.         </activity>  

在MainActivity接收数据:

  1. public class MainActivity extends Activity implements View.OnClickListener{  
  2.   
  3.     @Override  
  4.     protected void onCreate(Bundle savedInstanceState) {  
  5.         super.onCreate(savedInstanceState);  
  6.         setContentView(R.layout.activity_main);  
  7.   
  8.         Intent intent = getIntent();  
  9.         String scheme = intent.getScheme();  
  10.         Uri uri = intent.getData();  
  11.         System.out.println("scheme:"+scheme);  
  12.         if (uri != null) {  
  13.             String host = uri.getHost();  
  14.             String dataString = intent.getDataString();  
  15.             String id = uri.getQueryParameter("d");  
  16.             String path = uri.getPath();  
  17.             String path1 = uri.getEncodedPath();  
  18.             String queryString = uri.getQuery();  
  19.             System.out.println("host:"+host);  
  20.             System.out.println("dataString:"+dataString);  
  21.             System.out.println("id:"+id);  
  22.             System.out.println("path:"+path);  
  23.             System.out.println("path1:"+path1);  
  24.             System.out.println("queryString:"+queryString);  
  25.         }  
  26.           
  27.     }  
  28. }  


写一个最简单的网页:

  1. <a href="znn://aa.bb:80/test?p=12&d=1">test</a>  

测试结果:

scheme:znn
host:aa.bb
dataString:znn://aa.bb:80/test?p=12&id=1
id:1
path:/test
path1:/test

queryString:p=12&d=1


2.

如果我们想要匹配 http 以 “.pdf” 结尾的路径,使得别的程序想要打开网络 pdf 时,用户能够可以选择我们的程序进行下载查看。
我们可以将 scheme 设置为 “http”,pathPattern 设置为 “.*\\.pdf”,整个 intent-filter 设置为:
1 <intent-filter>
2 <action android:name="android.intent.action.VIEW"></action>
3 <category android:name="android.intent.category.DEFAULT"></category>
4 <data android:scheme="http" android:pathPattern=".*\\.pdf"></data>
5 </intent-filter>

如果你只想处理某个站点的 pdf,那么在 data 标签里增加  android:host="yoursite.com" 则只会匹配 http://yoursite.com/xxx/xxx.pdf,但这不会匹配 www.yoursite.com,如果你也想匹配这个站点的话,你就需要再添加一个 data 标签,除了 android:host 改为 “www.yoursite.com” 其他都一样。

3.

如果我们做的是一个音乐播放软件,当文件浏览器打开某音乐文件的时候,使我们的应用能够出现在选择框里?
 
这类似于文件关联了,其实做起来跟上面一样,也很简单,我们只用注册 android.intent.action.VIEW 与 mimeType 为 “audio/*” 就可以了,整个 intent-filter 设置为:
1 <intent-filter>
2 <action android:name="android.intent.action.VIEW" />
3 <category android:name="android.intent.category.DEFAULT" />
4 <data android:mimeType="audio/*" />
5 </intent-filter>



4.
首先在AndroidManifast.xml要被指定Scheme的Activity下设置如下参数

  1. <intent-filter>  
  2.           <category android:name="android.intent.category.DEFAULT"></category>  
  3.                 <action android:name="android.intent.action.VIEW"></action>  
  4.                 <data android:scheme="sh"></data>  
  5.             </intent-filter>  

这样即指定了接收Uri的Scheme为sh 且 Action为View的Intent。


利用如下Intent调用Activity

  1. startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("sh://123123123")));  



在接收的Activity中使用如下代码获得数据

  1. this.getIntent().getScheme();//获得Scheme名称  
  2. this.getIntent().getDataString();//获得Uri全部路径  





 实战:

1.被启动的ativity中data的配置:

<intent-filter>
    <data android:scheme="zy"  android:host="test"  android:port="80" />
    <category android:name="android.intent.category.DEFAULT" />
</intent-filter>
配置的时候,如果要配置host,必须先配置scheme,要配置port,必须先配置前面2个,就是scheme://host:port/path or pathPrefix or pathPattern 中要配置后面的属性,必须先配置前面的属性,不能有跨越


2.在另一个activity中启动上面的activity:

Intent itJump = new Intent();
itJump.setData(Uri.parse("zy://test:80"));
startActivity(itJump);

如果上面的data中配置到了port,这里也要写到port,如果是

itJump.setData(Uri.parse("zy://test"));

则匹配不上

阅读更多
文章标签: Android
个人分类: Android
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

不良信息举报

intent-filter 之 data 「scheme, host, port, mimeType, path, pathPrefix, pathPattern」

最多只允许输入30个字

加入CSDN,享受更精准的内容推荐,与500万程序员共同成长!
关闭
关闭