开发者社区> x-knight_勋爵> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

Android总结篇——Intent机制详解及示例总结

简介: 最近在进行android开发过程中,在将 Intent传递给调用的组件并完成组件的调用时遇到点困难,并且之前对Intent的学习也是一知半解,最近特意为此拿出一些时间,对Intent部分进行了系统的学习并进行了部分实践,下面将自己的学习及Intent知识进行了详细的归纳整理,希望能帮助到同样遇到相同问题的博友。
+关注继续查看

      最近在进行android开发过程中,在将 Intent传递给调用的组件并完成组件的调用时遇到点困难,并且之前对Intent的学习也是一知半解,最近特意为此拿出一些时间,对Intent部分进行了系统的学习并进行了部分实践,下面将自己的学习及Intent知识进行了详细的归纳整理,希望能帮助到同样遇到相同问题的博友。下面是Intent介绍、详解及Intent示例总结:

一.Intent介绍:

      Intent的中文意思是“意图,意向”,在Android中提供了Intent机制来协助应用间的交互与通讯,Intent负责对应用中一次操作的动 作、动作涉及数据、附加数据进行描述,Android则根据此Intent的描述,负责找到对应的组件,将 Intent传递给调用的组件,并完成组件的调用。Intent不仅可用于应用程序之间,也可用于应用程序内部的Activity/Service之间的 交互。因此,可以将Intent理解为不同组件之间通信的“媒介”专门提供组件互相调用的相关信息。

二、Intent作用:

      Intent 是一个将要执行的动作的抽象的描述,一般来说是作为参数来使用,由Intent来协助完成android各个组件之间的通讯。比如说调用 startActivity()来启动一个activity,或者由broadcaseIntent()来传递给所有感兴趣的 BroadcaseReceiver, 再或者由startService()/bindservice()来启动一个后台的service.所以可以看出来,intent主要是用来启动其他的 activity 或者service,所以可以将intent理解成activity之间的粘合剂。

 

三.Inten启动组件的方法:

     Intent可以启动一个Activity,也可以启动一个Service,还可以发起一个广播Broadcasts。具体方法如下:

组件名称

方法名称

 

Activity

startActvity( )

startActivity( )

 

Service

startService( )

bindService( )

 

Broadcasts

sendBroadcasts( )

sendOrderedBroadcasts( )

sendStickyBroadcasts( )

四.Intent的几个重要属性,下面进行详解:

      动作(Action),数据(Data),分类(Category),类型(Type),组件(Compent)以及扩展信(Extra)。其中最常用的是Action属性和Data属性。

1.Action属性:

      对于有如下声明的Activity

1 <activity android:name=".TargetActivity">  
2     <intent-filter>            
3         <action android:name="com.scott.intent.action.TARGET"/>  
4         <category android:name="android.intent.category.DEFAULT"/>  
5     </intent-filter>  
6 </activity>  

       TargetActivity在其<intent-filter>中声明了<action>,即目标action,如果我们需要做一个跳转的动作,就需要在Intent中指定目标的action,如下:

1 public void gotoTargetActivity(View view) {  
2     Intent intent = new Intent("com.scott.intent.action.TARGET");  
3     startActivity(intent);  
4 }  

       当我们为Intent指定相应的action,然后调用startActivity方法后,系统会根据action跳转到对应的Activity

       除了自定义的action之外,Intent也内含了很多默认的action,下面列举几个:

1 public static final String ACTION_MAIN = "android.intent.action.MAIN";  
2 public static final String ACTION_VIEW = "android.intent.action.VIEW";  
3 public static final String ACTION_WEB_SEARCH = "android.intent.action.WEB_SEARCH";  
4 public static final String ACTION_CALL = "android.intent.action.CALL";

      每一个action都有其特定的用途。

2.data和extras,即执行动作要操作的数据和传递到目标的附加信息:

     下面举一个与浏览器交互的例子:

 1     /** 
 2     * 打开指定网页 
 3     * @param view 
 4     */  
 5     public void invokeWebBrowser(View view) {  
 6     Intent intent = new Intent(Intent.ACTION_VIEW);  
 7     intent.setData(Uri.parse("http://www.google.com.hk"));  
 8     startActivity(intent);  
 9     }  
10     /** 
11     * 进行关键字搜索 
12     * @param view 
13     */  
14     public void invokeWebSearch(View view) {  
15     Intent intent = new Intent(Intent.ACTION_WEB_SEARCH);  
16     intent.putExtra(SearchManager.QUERY, "android");    //关键字  
17     startActivity(intent);  
18     }  

      上面两种方法分别是启动浏览器并打开指定网页、进行关键字搜索,分别对应的action是Intent.ACTION_VIEW和 Intent.ACTION_WEB_SEARCH,前者需指定相应的网页地址,后者需指定关键字信息,对于关键字搜索来说,浏览器会按照自己设置的默认 的搜索引擎进行搜索。

      我们注意到,在打开网页时,为Intent指定一个data属性,这其实是指定要操作的数据,是一个URI的形式,我们可以将一个指定前缀的字符串转换成 特定的URI类型,如:“http:”或“https:”表示网络地址类型,“tel:”表示电话号码类型,“mailto:”表示邮件地址类型,等等。

例如,我们要呼叫给定的号码,可以这样做:

1     public void call(View view) {  
2     Intent intent = new Intent(Intent.ACTION_CALL);  
3     intent.setData(Uri.parse("tel:12345678"));  
4     startActivity(intent);  
5     }  

      那么我们如何知道目标是否接受这种前缀呢?这就需要看一下目标中<data/>元素的匹配规则了。

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

android:scheme 匹配url中的前缀,除了“http”、“https”、“tel”...之外,我们可以定义自己的前缀

android:host 匹配url中的主机名部分,如“google.com”,如果定义为“*”则表示任意主机名

android:port 匹配url中的端口

android:path 匹配url中的路径

      我们改动一下TargetActivity的声明信息:

1     <activity android:name=".TargetActivity">  
2     <intent-filter>  
3     <action android:name="com.scott.intent.action.TARGET"/>  
4     <category android:name="android.intent.category.DEFAULT"/>  
5     <data android:scheme="scott" android:host="com.scott.intent.data" android:port="7788" android:path="/target"/>  
6     </intent-filter>  
7     </activity>  

  这个时候如果只指定action就不够了,我们需要为其设置data值,如下:

1     public void gotoTargetActivity(View view) {  
2     Intent intent = new Intent("com.scott.intent.action.TARGET");  
3     intent.setData(Uri.parse("scott://com.scott.intent.data:7788/target"));  
4     startActivity(intent);  
5     }  

  此时,url中的每个部分和TargetActivity配置信息中全部一致才能跳转成功,否则就被系统拒绝。

  不过有时候对path限定死了也不太好,比如我们有这样的url:(scott://com.scott.intent.data:7788/target/hello)(scott://com.scott.intent.data:7788/target/hi) 这个时候该怎么办呢?  我们需要使用另外一个元素:android:pathPrefix,表示路径前缀。     我们把android:path="/target"修改为android:pathPrefix="/target",然后就可以满足以上的要求了。 而在进行搜索时,我们使用了一个putExtra方法,将关键字做为参数放置在Intent中,我们成为extras(附加信息),这里面涉及到了一个Bundle对象。

       Bundle和Intent有着密不可分的关系,主要负责为Intent保存附加参数信息,它实现了android.os.Paracelable接口, 内部维护一个Map类型的属性,用于以键值对的形式存放附加参数信息。在我们使用Intent的putExtra方法放置附加信息时,该方法会检查默认的 Bundle实例为不为空,如果为空,则新创建一个Bundle实例,然后将具体的参数信息放置到Bundle实例中。我们也可以自己创建Bundle对 象,然后为Intent指定这个Bundle即可,如下:

1     public void gotoTargetActivity(View view) {  
2     Intent intent = new Intent("com.scott.intent.action.TARGET");  
3     Bundle bundle = new Bundle();  
4     bundle.putInt("id", 0);  
5     bundle.putString("name", "scott");  
6     intent.putExtras(bundle);  
7     startActivity(intent);  
8     }  

     需要注意的是,在使用putExtras方法设置Bundle对象之后,系统进行的不是引用操作,而是复制操作,所以如果设置完之后再更改bundle实 例中的数据,将不会影响Intent内部的附加信息。那我们如何获取设置在Intent中的附加信息呢?与之对应的是,我们要从Intent中获取到 Bundle实例,然后再从中取出对应的键值信息:

1     Bundle bundle = intent.getExtras();  
2     int id = bundle.getInt("id");  
3     String name = bundle.getString("name");  

       当然我们也可以使用Intent的getIntExtra和getStringExtra方法获取,其数据源都是Intent中的Bundle类型的实例对象。

3.category,要执行动作的目标所具有的特质或行为归类

      例如:在我们的应用主界面Activity通常有如下配置:

1 <category android:name="android.intent.category.LAUNCHER" />  

      代表该目标Activity是该应用所在task中的初始Activity并且出现在系统launcher的应用列表中。

      几个常见的category如下:

常量

解释

CATEGORY_DEFAULT

默认的category

CATEGORY_BROWSABLE

指定了此category后,在网页上点击图片或链接时,系统会考虑将此目标Activity列入可选列表,供用户选择以打开图片或链接。

CATEGORY_GADGET

The activity can be embedded inside of another activity that hosts gadgets.

CATEGORY_HOME

The activity displays the home screen, the first screen the user sees when the device is turned on or when the HOME key is pressed.

CATEGORY_LAUNCHER

The activity can be the initial activity of a task and is listed in the top-level application launcher.

CATEGORY_PREFERENCE

表示该目标Activity是一个首选项界面;

 

      在为Intent设置category时,应使用addCategory(String category)方法向Intent中添加指定的类别信息,来匹配声明了此类别的目标Activity。

下面举一个回到Home界面的例子

main.xml:

 1     <?xml version="1.0" encoding="utf-8"?> 
 2     <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
 3         android:orientation="vertical" android:layout_width="fill_parent" 
 4         android:layout_height="fill_parent" 
 5         >     
 6         <TextView   
 7         android:layout_width="fill_parent" 
 8         android:layout_height="wrap_content"   
 9         android:text="测试Intent Category"   
10         /> 
11         <Button   
12         android:id="@+id/Button1"   
13         android:layout_width="wrap_content" 
14         android:layout_height="wrap_content"   
15         android:text="转到Home界面" 
16         />    
17     </LinearLayout> 

strings.xml:

1     <?xml version="1.0" encoding="utf-8"?> 
2     <resources> 
3         <string name="hello">Hello World, MainActivity!</string> 
4         <string name="app_name">IntentCategoryDemo</string> 
5     </resources> 

MainActivity.java:

 1     package com.android.category.activity;  
 2      
 3     import android.app.Activity;  
 4     import android.content.Intent;  
 5     import android.os.Bundle;  
 6     import android.view.View;  
 7     import android.view.View.OnClickListener;  
 8     import android.widget.Button;  
 9      
10     public class MainActivity extends Activity {  
11         private Button btn;  
12         @Override 
13         public void onCreate(Bundle savedInstanceState) {  
14             super.onCreate(savedInstanceState);  
15             setContentView(R.layout.main);  
16               
17             btn = (Button)findViewById(R.id.Button1);  
18             btn.setOnClickListener(new OnClickListener() {  
19                 @Override 
20                 public void onClick(View v) {     
21                     Intent intent = new Intent();                 
22                     intent.setAction(Intent.ACTION_MAIN);// 添加Action属性                
23                     intent.addCategory(Intent.CATEGORY_HOME);// 添加Category属性              
24                     startActivity(intent);// 启动Activity  
25                 }  
26             });  
27         }  
28     } 

效果图如下:

https://yqfile.alicdn.com/img_5270146805d3e730b85ca135d0c8cb42.jpeg

Home:

https://yqfile.alicdn.com/img_8746df70f9e1646dcb722e0c80252911.jpeg

 

4.type:要执行动作的目标Activity所能处理的MIME数据类型

      例如:一个可以处理图片的目标Activity在其声明中包含这样的mimeType

1 <data android:mimeType="image/*" />  

      在使用Intent进行匹配时,我们可以使用setType(String type)或者setDataAndType(Uri data, String type)来设置mimeType。

5.component,目标组件的包或类名称

      在使用component进行匹配时,一般采用以下几种形式:

1 intent.setComponent(new ComponentName(getApplicationContext(), TargetActivity.class));  
2 intent.setComponent(new ComponentName(getApplicationContext(), "com.scott.intent.TargetActivity"));  
3 intent.setComponent(new ComponentName("com.scott.other", "com.scott.other.TargetActivity")); 

     其中,前两种是用于匹配同一包内的目标,第三种是用于匹配其他包内的目标。

    【注意】:如果我们在Intent中指定了component属性,系统将不会再对actiondata/typecategory进行匹配。

五、Intent用法示例全面总结:

1. 调用拨号程序

1 // 给移动客服10086拨打电话
2 Uri uri = Uri.parse("tel:10086");
3 Intent intent = new Intent(Intent.ACTION_DIAL, uri);
4 startActivity(intent);

2.发送短信或彩信

1 // 给10086发送内容为“Hello”的短信
2 Uri uri = Uri.parse("smsto:10086");
3 Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
4 intent.putExtra("sms_body", "Hello");
5 startActivity(intent);
1 // 发送彩信(相当于发送带附件的短信)
2 
3 Intent intent = new Intent(Intent.ACTION_SEND);
4 intent.putExtra("sms_body", "Hello");
5 Uri uri = Uri.parse("content://media/external/images/media/23");
6 intent.putExtra(Intent.EXTRA_STREAM, uri);
7 intent.setType("image/png");
8 startActivity(intent);

3.通过浏览器打开网页

1 // 打开Google主页
2 Uri uri = Uri.parse("http://www.google.com");
3 Intent intent = new Intent(Intent.ACTION_VIEW, uri);
4 startActivity(intent);

4.发送电子邮件

1 // 给someone@domain.com发邮件
2 Uri uri = Uri.parse("mailto:someone@domain.com");
3 Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
4 startActivity(intent);
1 // 给someone@domain.com发邮件发送内容为“Hello”的邮件
2 Intent intent = new Intent(Intent.ACTION_SEND);
3 intent.putExtra(Intent.EXTRA_EMAIL, "someone@domain.com");
4 intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
5 intent.putExtra(Intent.EXTRA_TEXT, "Hello");
6 intent.setType("text/plain");
7 startActivity(intent);
 1 // 给多人发邮件
 2 Intent intent=new Intent(Intent.ACTION_SEND);
 3 String[] tos = {"1@abc.com", "2@abc.com"}; // 收件人
 4 String[] ccs = {"3@abc.com", "4@abc.com"}; // 抄送
 5 String[] bccs = {"5@abc.com", "6@abc.com"}; // 密送
 6 intent.putExtra(Intent.EXTRA_EMAIL, tos);
 7 intent.putExtra(Intent.EXTRA_CC, ccs);
 8 intent.putExtra(Intent.EXTRA_BCC, bccs);
 9 intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
10 intent.putExtra(Intent.EXTRA_TEXT, "Hello");
11 intent.setType("message/rfc822");
12 startActivity(intent);

5.显示地图与路径规划

1 // 打开Google地图中国北京位置(北纬39.9,东经116.3)
2 Uri uri = Uri.parse("geo:39.9,116.3");
3 Intent intent = new Intent(Intent.ACTION_VIEW, uri);
4 startActivity(intent);
1 // 路径规划:从北京某地(北纬39.9,东经116.3)到上海某地(北纬31.2,东经121.4)
2 Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=39.9 116.3&daddr=31.2 121.4");
3 Intent intent = new Intent(Intent.ACTION_VIEW, uri);
4 startActivity(intent);

6. 播放多媒体

1 Intent intent = new Intent(Intent.ACTION_VIEW);
2 Uri uri = Uri.parse("file:///sdcard/foo.mp3");
3 intent.setDataAndType(uri, "audio/mp3");
4 startActivity(intent);
5 
6 Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
7 Intent intent = new Intent(Intent.ACTION_VIEW, uri);
8 startActivity(intent);

7. 拍照

1 // 打开拍照程序
2 Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
3 startActivityForResult(intent, 0);
1 // 取出照片数据
2 Bundle extras = intent.getExtras();
3 Bitmap bitmap = (Bitmap) extras.get("data");

8.获取并剪切图片

 1 // 获取并剪切图片
 2 Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
 3 intent.setType("image/*");
 4 intent.putExtra("crop", "true"); // 开启剪切
 5 intent.putExtra("aspectX", 1); // 剪切的宽高比为1:2
 6 intent.putExtra("aspectY", 2);
 7 intent.putExtra("outputX", 20); // 保存图片的宽和高
 8 intent.putExtra("outputY", 40);
 9 intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp"))); // 保存路径
10 intent.putExtra("outputFormat", "JPEG");// 返回格式
11 startActivityForResult(intent, 0);
 1 // 剪切特定图片
 2 
 3  1 Intent intent = new Intent("com.android.camera.action.CROP");
 4  2 intent.setClassName("com.android.camera", "com.android.camera.CropImage");
 5  3 intent.setData(Uri.fromFile(new File("/mnt/sdcard/temp")));
 6  4 intent.putExtra("outputX", 1); // 剪切的宽高比为1:2
 7  5 intent.putExtra("outputY", 2);
 8  6 intent.putExtra("aspectX", 20); // 保存图片的宽和高
 9  7 intent.putExtra("aspectY", 40);
10  8 intent.putExtra("scale", true);
11  9 intent.putExtra("noFaceDetection", true);
12 10 intent.putExtra("output", Uri.parse("file:///mnt/sdcard/temp"));
13 11 startActivityForResult(intent, 0);

9. 打开Google Market

1 // 打开Google Market直接进入该程序的详细页面
2 Uri uri = Uri.parse("market://details?id=" + "com.demo.app");
3 Intent intent = new Intent(Intent.ACTION_VIEW, uri);
4 startActivity(intent);

10.安装和卸载程序

1 Uri uri = Uri.fromParts("package", "com.demo.app", null);
2 Intent intent = new Intent(Intent.ACTION_DELETE, uri);
3 startActivity(intent);

11. 进入设置界面

1 // 进入无线网络设置界面(其它可以举一反三)
2 Intent intent = new Intent(android.provider.Settings.ACTION_WIRELESS_SETTINGS);
3 startActivityForResult(intent, 0);

 

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
Android应用内广播LocalBroadcastManager机制详解
终于建了一个自己个人小站:https://huangtianyu.gitee.io,以后优先更新小站博客,欢迎进站,O(∩_∩)O~~ 1. 简介 通常我们在使用Android广播的时候都会直接将广播注册到系统的AMS当中,由于AMS任务繁忙,一般可能不会立即能处理到我们发出的广播,如果我们使用广播是在应用内的单个进程中使用,则完全可以采用LocalBroadcastManager来处理。
1123 0
【Android 应用开发】Android开发技巧--Application, ListView排列,格式化浮点数,string.xml占位符,动态引用图片
【Android 应用开发】Android开发技巧--Application, ListView排列,格式化浮点数,string.xml占位符,动态引用图片
151 0
Android大数据、断点续传、耗时下载之DownloadManager开发简介(1)
 Android大数据、断点续传、耗时下载之DownloadManager开发简介(1) Android涉及到的网络数据请求,如果是零星数据、且数据量较小(几十KB到几百KB,1MB以内),一般的,可以自己使用Android原生HTTP或者第三方开源框架如Volley(相关文章:http://blog.
933 0
Android开发重要参考资料
=======================博客============================= 秋百万 有心课堂 郭霖 源码 安装ffmpeg 胡凯 官方培训课程 litesuitsway 爱哥 trinea robinRobin Hu...
848 0
+关注
x-knight_勋爵
在读普通小硕,本硕都是计算机不科学专业; 热衷于任何新兴技术及事物; 对区块链拥有信仰,毕业后 All in blockchain !
文章
问答
文章排行榜
最热
最新
相关电子书
更多
Android应用启动速度和内存优化实践
立即下载
Android组件化实现
立即下载
Android内存泄漏自动化链路分析组件Probe
立即下载