Android Activity和Intent机制学习笔记

移动开发 Android
本文介绍了 Android 应用编程中 Activity 的生命周期、通信方式和 Intent Filter 等内容,并提供了一些日常开发中经常用到的关于 Activity 的技巧和方法。通过本文,你可以进一步了接 Android 中 Activity 的运作方式。

Activity

Android中,Activity是所有程序的根本,所有程序的流程都运行在Activity之中,Activity具有自己的生命周期(见http://www.cnblogs.com/feisky/archive/2010/01/01/1637427.html,由系统控制生命周期,程序无法改变,但可以用onSaveInstanceState保存其状态)。

对于Activity,关键是其生命周期的把握(如下图),其次就是状态的保存和恢复(onSaveInstanceState onRestoreInstanceState),以及Activity之间的跳转和数据传输(intent)。

Activity中常用的函数有SetContentView()   findViewById()    finish()   startActivity(),其生命周期涉及的函数有:

  1. void onCreate(Bundle savedInstanceState)  
  2. void onStart()  
  3. void onRestart()  
  4. void onResume()  
  5. void onPause()  
  6. void onStop()  
  7. void onDestroy() 

注意的是,Activity的使用需要在Manifest文件中添加相应的<Activity>,并设置其属性和intent-filter。

Intent

Android中提供了Intent机制来协助应用间的交互与通讯,Intent负责对应用中一次操作的动作、动作涉及数据、附加数据进行描述,Android则根据此Intent的描述,负责找到对应的组件,将 Intent传递给调用的组件,并完成组件的调用。Intent不仅可用于应用程序之间,也可用于应用程序内部的Activity/Service之间的交互。因此,Intent在这里起着一个媒体中介的作用,专门提供组件互相调用的相关信息,实现调用者与被调用者之间的解耦。在SDK中给出了Intent作用的表现形式为:

通过Context.startActivity() orActivity.startActivityForResult() 启动一个Activity; 
通过 Context.startService() 启动一个服务,或者通过Context.bindService() 和后台服务交互; 
通过广播方法(比如 Context.sendBroadcast(),Context.sendOrderedBroadcast(),  Context.sendStickyBroadcast()) 发给broadcast receivers。
Intent属性的设置,包括以下几点:(以下为XML中定义,当然也可以通过Intent类的方法来获取和设置)

(1)Action,也就是要执行的动作

SDk中定义了一些标准的动作,包括

 

当然,也可以自定义动作(自定义的动作在使用时,需要加上包名作为前缀,如"com.example.project.SHOW_COLOR”),并可定义相应的Activity来处理我们的自定义动作。

(2)Data,也就是执行动作要操作的数据

Android中采用指向数据的一个URI来表示,如在联系人应用中,一个指向某联系人的URI可能为:content://contacts/1。对于不同的动作,其URI数据的类型是不同的(可以设置type属性指定特定类型数据),如ACTION_EDIT指定Data为文件URI,打电话为tel:URI,访问网络为http:URI,而由content provider提供的数据则为content: URIs。

(3)type(数据类型),显式指定Intent的数据类型(MIME)。一般Intent的数据类型能够根据数据本身进行判定,但是通过设置这个属性,可以强制采用显式指定的类型而不再进行推导。

(4)category(类别),被执行动作的附加信息。例如 LAUNCHER_CATEGORY 表示Intent 的接受者应该在Launcher中作为顶级应用出现;而ALTERNATIVE_CATEGORY表示当前的Intent是一系列的可选动作中的一个,这些动作可以在同一块数据上执行。还有其他的为

Constant Meaning 

(5)component(组件),指定Intent的的目标组件的类名称。通常 Android会根据Intent 中包含的其它属性的信息,比如action、data/type、category进行查找,最终找到一个与之匹配的目标组件。但是,如果 component这个属性有指定的话,将直接使用它指定的组件,而不再执行上述查找过程。指定了这个属性以后,Intent的其它所有属性都是可选的。

(6)extras(附加信息),是其它所有附加信息的集合。使用extras可以为组件提供扩展信息,比如,如果要执行“发送电子邮件”这个动作,可以将电子邮件的标题、正文等保存在extras里,传给电子邮件发送组件。

理解Intent的关键之一是理解清楚Intent的两种基本用法:一种是显式的Intent,即在构造Intent对象时就指定接收者;另一种是隐式的Intent,即Intent的发送者在构造Intent对象时,并不知道也不关心接收者是谁,有利于降低发送者和接收者之间的耦合。

对于显式Intent,Android不需要去做解析,因为目标组件已经很明确,Android需要解析的是那些隐式Intent,通过解析,将 Intent映射给可以处理此Intent的Activity、IntentReceiver或Service。        

Intent解析机制主要是通过查找已注册在AndroidManifest.xml中的所有IntentFilter及其中定义的Intent,最终找到匹配的Intent。在这个解析过程中,Android是通过Intent的action、type、category这三个属性来进行判断的,判断方法如下:

如果Intent指明定了action,则目标组件的IntentFilter的action列表中就必须包含有这个action,否则不能匹配; 
如果Intent没有提供type,系统将从data中得到数据类型。和action一样,目标组件的数据类型列表中必须包含Intent的数据类型,否则不能匹配。 
如果Intent中的数据不是content: 类型的URI,而且Intent也没有明确指定它的type,将根据Intent中数据的scheme (比如 http: 或者mailto:) 进行匹配。同上,Intent 的scheme必须出现在目标组件的scheme列表中。 
如果Intent指定了一个或多个category,这些类别必须全部出现在组建的类别列表中。比如Intent中包含了两个类别:LAUNCHER_CATEGORY 和 ALTERNATIVE_CATEGORY,解析得到的目标组件必须至少包含这两个类别。
Intent-Filter的定义

一些属性设置的例子:

  1. <action android:name="com.example.project.SHOW_CURRENT" /> 
  2. <category android:name="android.intent.category.DEFAULT" /> 
  3. <data android:mimeType="video/mpeg" android:scheme="http" . . . />   
  4. <data android:mimeType="image/*" /> 
  5. <data android:scheme="http" android:type="video/*" /> 

完整的实例

  1. <activity android:name="NotesList" android:label="@string/title_notes_list">   
  2. <intent-filter>                  
  3. <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> 
  4. </intent-filter>   
  5. <intent-filter>          
  6. <action android:name="android.intent.action.VIEW" /> 
  7. <action android:name="android.intent.action.EDIT" />     <action android:name="android.intent.action.PICK" />     <category 
  8. android:name="android.intent.category.DEFAULT" />   
  9. <data android:mimeType="vnd.android.cursor.dir/vnd.google.note" />   
  10. </intent-filter>              
  11. <intent-filter>                  
  12. <action android:name="android.intent.action.GET_CONTENT" />      <category 
  13. android:name="android.intent.category.DEFAULT" />        <data android:mimeType="vnd.android.cursor.item/vnd.google.note" />              
  14. </intent-filter>         
  15. </activity> 

#p#

Intent用法实例

1.无参数Activity跳转

  1. Intent it = new Intent(Activity.Main.this, Activity2.class);  
  2. startActivity(it);    

 
2.向下一个Activity传递数据(使用Bundle和Intent.putExtras)

  1. Intent it = new Intent(Activity.Main.this, Activity2.class);  
  2. Bundle bundle=new Bundle();  
  3. bundle.putString("name", "This is from MainActivity!");  
  4. it.putExtras(bundle);       // it.putExtra(“test”, "shuju”);  
  5. startActivity(it);            // startActivityForResult(it,REQUEST_CODE); 

 
对于数据的获取可以采用:

  1. Bundle bundle=getIntent().getExtras();  
  2. String name=bundle.getString("name"); 

 
3.向上一个Activity返回结果(使用setResult,针对startActivityForResult(it,REQUEST_CODE)启动的Activity)

 

  1. Bundle bundle=getIntent().getExtras();  
  2. String name=bundle.getString("name"); 

 

4.回调上一个Activity的结果处理函数(onActivityResult)

  1. @Override  
  2.     protected void onActivityResult(int requestCode, int resultCode, Intent data) {  
  3.         // TODO Auto-generated method stub  
  4.         super.onActivityResult(requestCode, resultCode, data);  
  5.         if (requestCode==REQUEST_CODE){  
  6.             if(resultCode==RESULT_CANCELED)  
  7.                   setTitle("cancle");  
  8.             else if (resultCode==RESULT_OK) {  
  9.                  String temp=null;  
  10.                  Bundle bundle=data.getExtras();  
  11.                  if(bundle!=null)   temp=bundle.getString("name");  
  12.                  setTitle(temp);  
  13.             }  
  14.         }  
  15.     } 

下面是转载来的其他的一些Intent用法实例(转自javaeye)

显示网页

  1. 1. Uri uri = Uri.parse("http://google.com");    
  2. 2. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  3. 3. startActivity(it);  

显示地图

  1. Uri uri = Uri.parse("http://google.com");    
  2. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  3. startActivity(it);  

 

路径规划

  1. Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=startLat%20startLng&daddr=endLat%20endLng&hl=en");    
  2. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  3.  startActivity(it);    
  4. //where startLat, startLng, endLat, endLng are a long with 6 decimals like: 50.123456   

打电话

  1. //叫出拨号程序   
  2. Uri uri = Uri.parse("tel:0800000123");    
  3. Intent it = new Intent(Intent.ACTION_DIAL, uri);    
  4. startActivity(it);   

//直接打电话出去  

  1. Uri uri = Uri.parse("tel:0800000123");    
  2. Intent it = new Intent(Intent.ACTION_CALL, uri);    
  3. startActivity(it);    
  4. //用這個,要在 AndroidManifest.xml 中,加上    
  5. //<uses-permission id="android.permission.CALL_PHONE" />   

传送SMS/MMS

  1. //调用短信程序   
  2. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  3. it.putExtra("sms_body", "The SMS text");     
  4. it.setType("vnd.android-dir/mms-sms");    
  5. startActivity(it);   
  6. //传送消息   
  7.  Uri uri = Uri.parse("smsto://0800000123");    
  8. Intent it = new Intent(Intent.ACTION_SENDTO, uri);    
  9. it.putExtra("sms_body", "The SMS text");    
  10. startActivity(it);   
  11. //传送 MMS    
  12. Uri uri = Uri.parse("content://media/external/images/media/23");    
  13. Intent it = new Intent(Intent.ACTION_SEND);     
  14. it.putExtra("sms_body", "some text");     
  15. it.putExtra(Intent.EXTRA_STREAM, uri);    
  16. it.setType("image/png");     
  17. startActivity(it);   

传送 Email

  1. Uri uri = Uri.parse("mailto:xxx@abc.com");    
  2. Intent it = new Intent(Intent.ACTION_SENDTO, uri);    
  3. startActivity(it);    
  4.  
  5. Intent it = new Intent(Intent.ACTION_SEND);    
  6. it.putExtra(Intent.EXTRA_EMAIL, "me@abc.com");    
  7. it.putExtra(Intent.EXTRA_TEXT, "The email body text");    
  8. it.setType("text/plain");    
  9. startActivity(Intent.createChooser(it, "Choose Email Client"));    
  10.  
  11. Intent it=new Intent(Intent.ACTION_SEND);      
  12. String[] tos={"me@abc.com"};      
  13. String[] ccs={"you@abc.com"};      
  14. it.putExtra(Intent.EXTRA_EMAIL, tos);      
  15. it.putExtra(Intent.EXTRA_CC, ccs);      
  16. it.putExtra(Intent.EXTRA_TEXT, "The email body text");      
  17. it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");      
  18. it.setType("message/rfc822");      
  19. startActivity(Intent.createChooser(it, "Choose Email Client"));   
  20. //传送附件  
  21. Intent it = new Intent(Intent.ACTION_SEND);    
  22.  it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");    
  23. it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/mysong.mp3");    
  24. sendIntent.setType("audio/mp3");    
  25. startActivity(Intent.createChooser(it, "Choose Email Client"));  

播放多媒体

  1. Uri uri = Uri.parse("file:///sdcard/song.mp3");    
  2. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  3. it.setType("audio/mp3");    
  4. startActivity(it);   
  5.  Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");    
  6. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  7. startActivity(it);   
  8.  
  9. Market 相关  
  10. //寻找某个应用   
  11.  Uri uri = Uri.parse("market://search?q=pname:pkg_name");   
  12. Intent it = new Intent(Intent.ACTION_VIEW, uri);   startActivity(it);    
  13. //where pkg_name is the full package path for an application   
  14. /显示某个应用的相关信息   
  15. Uri uri = Uri.parse("market://details?id=app_id");    
  16. Intent it = new Intent(Intent.ACTION_VIEW, uri);  startActivity(it);    
  17. /where app_id is the application ID, find the ID     
  18. //by clicking on your application on Market home     
  19. //page, and notice the ID from the address bar  

Uninstall 应用程序

  1. Uri uri = Uri.parse("file:///sdcard/song.mp3");    
  2. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  3. it.setType("audio/mp3");    
  4. startActivity(it);   
  5.  Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");    
  6. Intent it = new Intent(Intent.ACTION_VIEW, uri);    
  7. startActivity(it);   
  8. Market 相关  
  9. //寻找某个应用   
  10.  Uri uri = Uri.parse("market://search?q=pname:pkg_name");   
  11. Intent it = new Intent(Intent.ACTION_VIEW, uri);   startActivity(it);    
  12. //where pkg_name is the full package path for an application   
  13. /显示某个应用的相关信息   
  14. Uri uri = Uri.parse("market://details?id=app_id");    
  15. Intent it = new Intent(Intent.ACTION_VIEW, uri);  startActivity(it);    
  16. /where app_id is the application ID, find the ID     
  17. //by clicking on your application on Market home     
  18. //page, and notice the ID from the address bar  

【编辑推荐】

责任编辑:zhaolei 来源: 网络转载
相关推荐

2014-07-15 10:16:02

AndroidIntent

2011-04-12 17:16:52

Activity跳转开发实例Android学习笔记

2013-03-28 09:07:37

Android开发Intent机制

2013-01-10 14:54:48

Android开发组件Intent

2013-05-27 14:06:14

Android开发移动开发Intent机制

2014-08-08 10:36:12

ActivityAndroid

2014-08-21 11:07:40

AndroidActivity销毁

2010-09-26 16:55:31

JVM学习笔记

2010-06-30 10:30:29

UML动态建模

2011-06-03 13:11:19

Android Activity

2009-04-03 08:26:02

2010-08-18 10:13:55

IntentAndroid

2013-12-26 16:24:13

Android开发Android应用Intents

2010-01-25 16:52:22

Android Int

2013-01-10 15:36:44

Android开发组件Intent

2011-09-09 13:59:17

Android wid

2010-02-07 10:40:55

Android Int

2009-04-03 08:21:37

AndroidGoogle移动OS

2013-01-08 13:33:07

Android开发Activity入门指南

2012-02-17 17:07:30

Android安全Activity劫持
点赞
收藏

51CTO技术栈公众号