Android开发实践:实战演练隐式Intent的用法

简介:

任务假设我们已经实现了一个视频播放器(PlayerActivity),我们希望能把它注册到系统中,当用户点击本地视频或者在线视频时,能启动这个视频播放器。


假设该类的全路径为:com.jhuster.videoplayer.PlayerActivity)


[注]:本文完整的示例代码请到我的Github下载,地址:VideoPlayer


1. 什么是隐式Intent?


Intent是Android中比较重要的组件,常用来启动一个新的Activity或者Service、广播某个事件,以及在Android组件之间传递数据。通过Intent来启动新的Activity或者Service通常有两种方法,一种是显示启动,另一种是隐式启动。


显示启动就是在明确指出要启动的Activity或者Service的类或者包名。例如:


1
2
3
4
5
6
7
8
9
10
Intent intent = newIntent( this , PlayerActivity. class );  
startActivity(intent);
  
Intent intent =  new  Intent();
intent.setClass( this ,PlayerActivity. class );  
startActivity(intent);  
  
Intent intent =  new  Intent();
intent.setClassName(“com.jhuster.videoplayer”,“com.jhuster.videoplayer.PlayerActivity”); 
startActivity(intent);


隐式启动则是不明确指定启动哪个Activity或者Service,而是通过设置Action、Data、Category,让系统来筛选出合适的目标。


例如拨打电话: 


1
2
Intent intent =  new  Intent(Intent.ACTION_DIAL,Uri.parse(“tel: 021 - 80961111 ”));
startActivity(intent);


系统接收到隐式启动请求后,会根据系统中各个Activity在AndroidManifest.xml文件中声明的<intent-filter>来比较和判断是否匹配当前的Intent请求的。


因此,如果我们希望PlayerActivity能够被系统隐式启动,则首先需要在AndroidManifest.xml文件中为该Activity添加<intent-filter>.


2. 为PlayerActivity添加<intent-filter>


<intent-filter>的标签有很多,这里只介绍和添加最基本且最常用的三个标签,分别是<action>,<category>和<data>。


2.1 添加<action>


这个标签是必须添加的,可以自己定义,也可以使用系统预定义的变量,Android系统默认定义了很多action,具体可以查看SDK文档,或者Google一下“android.intent.action.”。


这里,因为我们的类是用来“播放视频”的,因此可以使用系统预定义的:android.intent.action.VIEW,它表示需要启动某个Activity显示指定的数据(包括图片、视频、文档等)。


添加了<action>后的<activity>如下所示:


1
2
3
4
5
<activity android:name= "com.jhuster.videoplayer.PlayerActivity" >
     <intent-filter>
         <action android:name= "android.intent.action.VIEW"  />        
     </intent-filter>            
</activity>


2.2 添加<category>


category代表类别,定义了Activity的类别,Activity可以设置一个或者多个category标签。常用的一般有3个:DEFAULT,HOME,LAUNCHER


1
2
3
DEFAULT  默认动作
HOME     设置为本地桌面应用
LAUNCHER 本APP的启动Activity


本应用中我们使用DEFAULT类别即可,DEFAULT也是category最常用的选项。


添加了category后的<activity>如下所示:


1
2
3
4
5
6
<activityandroid:name= "com.jhuster.videoplayer.PlayerActivity" >
     <intent-filter>
         <actionandroid:name= "android.intent.action.VIEW"  />        
         <categoryandroid:name= "android.intent.category.DEFAULT"  />
     </intent-filter>            
</activity>


2.3 添加<data>


data 代表数据源,是<intent-filter>中最复杂的标签,因为不同的Activity支持的数据来源和类型多种多样,所以需要通过详细的data标签信息来指明。


data 标签有很多属性,包括:

1
2
3
4
5
android:host: 指定主机名,例如:google.com
android:port:  制定主机端口,例如: 80
android:path:  指定URL的有效路径值,例如: /index/examples
android:mimeType: 指定组件可以执行的数据类型,例如:image/jpeg,video/*
android:scheme: 指定特定的模式,例如:content,http


这里,假设我们的视频播放器支持多种数据来源,包括:本地视频文件,本地媒体URL,网络视频流(HTTP、RTMP、RTSP协议),另外,假设我们的视频播放器只支持mp4和3gpp两种文件格式。


那么,下面我们来添加两种最常用的<data>标签,scheme和mimeType,并且解释每条标签对应的是怎样的一种数据来源或者数据格式。


(1) <data android:scheme="xxx"/>


这里的xxx可以是:file,content,网络协议(HTTP,RTMP、RTSP等)


本应用中我们给PlayerActivity的<Intent-filter>中添加:


1
2
3
4
< data  android:scheme = "file" />
< data  android:scheme = "content" />
< data  android:scheme = "http" />
< data  android:scheme = "rtsp" />


添加了这样几条data标签项之后,如果隐式Intent中的数据来源URL是以“file://”、“content://”、“http://”、“rtsp://”开头的URL资源,都会隐式地启动我们的PlayerActivity。


例如,其他的Activity可以通过下面的方法来隐式启动我们的PlayerActivity.


1
2
3
Intent intent =  new  Intent(Intent.ACTION_VIEW);        
intent.setData(Uri.fromFile( new  File( "/sdcard/test.3gp" )));
startActivity(intent);


Uri.fromFile这条语句会把指定的文件位置转换为以“file://”开头的Uri对象,如上述例子最终得到的URL为:“file:///sdcard/test.3gp”


同理,可以通过Uri.parse来转换我们常见的网络地址字符串为Uri对象,例如:


1
2
3
Intent intent =  new  Intent(Intent.ACTION_VIEW);        
intent.setData(Uri.parse( "http://ticktick.blog.51cto.com/test.mp4" ));
startActivity(intent);


(2) <data android:mimeType="xxx"/>


mimeType用来设置数据类型,例如图像数据(image/png或者image/*),视频数据(video/mp4或者video/*),如果使用*代表匹配所有的子类型。


MIME TYPE是互联网的一种标记数据类型的标准,现在已经支持非常多的类型了,这里我不一一列举,大家可以在Google上搜索一下。


本应用中我们假设需要支持的是mp4和3gpp两种类型,那么,我们可以添加这样两条 mimeType :


1
2
<data android:mimeType= "video/3gpp" />
<data android:mimeType= "video/mp4"  />


那么,其他的Activity就可以通过下面的方法来隐式启动我们的PlayerActivity. 注意,当<Intent-filter>已经添加了mimeType之后,隐式Intent必须设置Type参数才能匹配到该Activity,所以建议使用setDataAndType方法,而不是单一的setData方法。


1
2
3
Intent intent =  new  Intent(Intent.ACTION_VIEW);        
intent.setDataAndType(Uri.fromFile( new  File( "/sdcard/test.3gp" )), "video/3gpp" );
startActivity(intent);


当然,这里的"video/3gpp"也可以写成:"video/*",但这样可能会匹配到一些不支持3gpp的播放器。


(3) 小结


添加了<data>标签后的<intent-filter>如下所示:


1
2
3
4
5
6
7
8
9
10
11
12
13
< activity  android:name = "com.jhuster.videoplayer.PlayerActivity" >
     < intent-filter >
         < action  android:name = "android.intent.action.VIEW"  />
         < category  android:name = "android.intent.category.DEFAULT"  />
         < data  android:scheme = "file" />
         < data  android:scheme = "content" />
         < data  android:scheme = "http" />
         < data  android:scheme = "rtsp" />
         < data  android:scheme = "rtmp" />                
         < data  android:mimeType = "video/3gpp"  />
         < data  android:mimeType = "video/mp4"  />
     </ intent-filter >            
</ activity >


3.  在PlayerActivity中获取参数


通过上面的介绍,我们已经知道了怎样添加<intent-filter>以及怎样通过隐式Intent来调用我们的PlayerActivity,那么,下面我们还要了解如何在PlayerActivity中解析来自隐式Intent的参数


其实,Intent提供了很多方法可以Get相关的参数信息,例如:


1
2
3
4
public  String getAction();
public  Uri    getData();
public  String getScheme();
public  String getType();


上述方法分别可以获取Intent的Action,Data Uri,Scheme和MimeType值。


对于“file://”开头的Uri对象,我们可以通过Uri.getPath方法得到去除了“file://”前缀的具体文件地址。例如: “file:///sdcard/test.mp4”则可以转换为实际的“/sdcard/test.mp4”。


对于网络码流,例如:“http://”、“rtsp://”等开头的Uri,则可以直接通过toString()方法转换为实际地址的字符串。


而对于“content://”开头的URI对象,一般是从系统的媒体数据库中检索出来的结果,因此需要反向查找得到实际的文件地址,这里提供一个函数进行转换。


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public  static  String getVideoPath(Context context, Uri uri) {
         
     Uri videopathURI = uri;
     if  (uri.getScheme().toString().compareTo( "content" ) ==  0  ) {      
         Cursor cursor = context.getContentResolver().query(uri,  null null null null );
         if  (cursor.moveToFirst()) {
             int  column_index = cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DATA);
             videopathURI = Uri.parse(cursor.getString(column_index));
             return  videopathURI.getPath();
         }
     }
     else  if  (uri.getScheme().compareTo( "file" ) ==  0  ) {
         return  videopathURI.getPath();
     }
     
     return  videopathURI.toString();
}


本文转自 Jhuster 51CTO博客,原文链接:http://blog.51cto.com/ticktick/1621957,如需转载请自行联系原作者

相关文章
|
11天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
16天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
2天前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
18天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异和挑战
【10月更文挑战第37天】在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统扮演着主角。它们各自拥有独特的特性、优势以及面临的开发挑战。本文将深入探讨这两个平台在开发过程中的主要差异,从编程语言到用户界面设计,再到市场分布的不同影响,旨在为开发者提供一个全面的视角,帮助他们更好地理解并应对在不同平台上进行应用开发时可能遇到的难题和机遇。
|
20天前
|
XML 存储 Java
探索安卓开发之旅:从新手到专家
【10月更文挑战第35天】在数字化时代,安卓应用的开发成为了一个热门话题。本文旨在通过浅显易懂的语言,带领初学者了解安卓开发的基础知识,同时为有一定经验的开发者提供进阶技巧。我们将一起探讨如何从零开始构建第一个安卓应用,并逐步深入到性能优化和高级功能的实现。无论你是编程新手还是希望提升技能的开发者,这篇文章都将为你提供有价值的指导和灵感。
|
18天前
|
存储 API 开发工具
探索安卓开发:从基础到进阶
【10月更文挑战第37天】在这篇文章中,我们将一起探索安卓开发的奥秘。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和建议。我们将从安卓开发的基础开始,逐步深入到更复杂的主题,如自定义组件、性能优化等。最后,我们将通过一个代码示例来展示如何实现一个简单的安卓应用。让我们一起开始吧!
|
19天前
|
存储 XML JSON
探索安卓开发:从新手到专家的旅程
【10月更文挑战第36天】在这篇文章中,我们将一起踏上一段激动人心的旅程,从零基础开始,逐步深入安卓开发的奥秘。无论你是编程新手,还是希望扩展技能的老手,这里都有适合你的知识宝藏等待发掘。通过实际的代码示例和深入浅出的解释,我们将解锁安卓开发的关键技能,让你能够构建自己的应用程序,甚至贡献于开源社区。准备好了吗?让我们开始吧!
26 2
|
20天前
|
Android开发
布谷语音软件开发:android端语音软件搭建开发教程
语音软件搭建android端语音软件开发教程!
|
25天前
|
JSON Java Android开发
探索安卓开发之旅:打造你的第一个天气应用
【10月更文挑战第30天】在这个数字时代,掌握移动应用开发技能无疑是进入IT行业的敲门砖。本文将引导你开启安卓开发的奇妙之旅,通过构建一个简易的天气应用来实践你的编程技能。无论你是初学者还是有一定经验的开发者,这篇文章都将成为你宝贵的学习资源。我们将一步步地深入到安卓开发的世界中,从搭建开发环境到实现核心功能,每个环节都充满了发现和创造的乐趣。让我们开始吧,一起在代码的海洋中航行!
|
20天前
|
前端开发 Android开发 UED
安卓应用开发中的自定义控件实践
【10月更文挑战第35天】在移动应用开发中,自定义控件是提升用户体验、增强界面表现力的重要手段。本文将通过一个安卓自定义控件的创建过程,展示如何从零开始构建一个具有交互功能的自定义视图。我们将探索关键概念和步骤,包括继承View类、处理测量与布局、绘制以及事件处理。最终,我们将实现一个简单的圆形进度条,并分析其性能优化。