参考《Professional Android 4 Development》
Intents
本文主要从这几个方面介绍Intents:
- 使用Intents在Android程序内部或外部通信
- 监听Intents,从而获取系统状态的变化
- 使用Implicit Intents或Explicit Intents来启动Activity或Service
- Broadcast Intents
Intents简介
Android使用Intents进行消息传递,你可以使用Intents做这些事情:
- 启动Service或Activity
- 启动并传数据给Service或Activity
- 广播事件发生消息
使用Intents,你可以启动Android中任意一个应用中的任意一个component。Intents也可以用于发送广播消息,通过在应用程序中注册Broadcast Receivers,就可以接收相应的广播消息。
使用Intents启动Activity
调用startActivity(Intents)方法(参数为Intents对象),可以启动Activity。根据Intents的不同,又可以分为两种方式:一是在Intents中指定要启动的Activity类名称,从而启动指定的Activity;另一种是在Intents中指定要做的动作,由系统来寻找最佳的Activity来完成这个Intents。
startActivity()不会返回执行结果。若要获得Activity的执行结果,使用 startActivityForResult方法。
在Intents中指定的Activity:
Intent intent = new Intent(MyActivity.this, MyOtherActivity.class); startActivity(intent);
在Intents中指定要执行的动作:
Intent intent = new Intent(Intent.ACTION_DIAL, Uri.parse(“tel:555-2368”)); startActivity(intent);
这种只在指定动作而不是具体的Activity类的startActivity()的Intents也叫Implicit Intents。Implicit Intents的构造函数中包含要执行的动作,有时也会传数据参数。使用putExtra()方法可以传更多的参数进去,获取这些参数的方法则为get[type]Extra()。
测试Intents能否被解析
对于Implicit Intents,我们无法在开发的时候确定系统中能否解析这个Intents。因此,在运行时测试Intent能否被解析就很重要。使用PackManager和Intent中的resolveIntent()方法,可以测试Intent能否被解析:
if (somethingWeird && itDontLookGood) { // Create the impliciy Intent to use to start a new Activity. Intent intent = new Intent(Intent.ACTION_DIAL, Uri.parse(“tel:555-2368”)); // Check if an Activity exists to perform this action. PackageManager pm = getPackageManager(); ComponentName cn = intent.resolveActivity(pm); if (cn == null) { // If there is no Activity available to perform the action // Check to see if the Google Play Store is available. Uri marketUri = Uri.parse(“market://search?q=pname:com.myapp.packagename”); Intent marketIntent = new Intent(Intent.ACTION_VIEW).setData(marketUri); // If the Google Play Store is available, use it to download an application capable of performing the required action. Otherwise log an error. if (marketIntent.resolveActivity(pm) != null) startActivity(marketIntent); else Log.d(TAG, “Market client not available.”); }else startActivity(intent); }
获取从Activity中返回的结果
使用startActivityForResult方法启动的Activity会产生返回值,这种Activity也被成为sub Activity。与startActivity方法相比,startActivityForResult方法多了一个request code参数,后面讲使用这个request code获取返回值。startActivityForResult示例代码如下:
private static final int SHOW_SUBACTIVITY = 1; private void startSubActivity() { Intent intent = new Intent(this, MyOtherActivity.class); startActivityForResult(intent, SHOW_SUBACTIVITY); }
为了获取sub activity返回的数据,在sub activity结束前要调用setResult()和finish()方法,例如:
Button okButton = (Button) findViewById(R.id.ok_button); okButton.setOnClickListener(new View.OnClickListener() { public void onClick(View view) { long selected_horse_id = listView.getSelectedItemId(); Uri selectedHorse = Uri.parse(“content://horses/” + selected_horse_id); Intent result = new Intent(Intent.ACTION_PICK, selectedHorse); setResult(RESULT_OK, result); finish(); } }); Button cancelButton = (Button) findViewById(R.id.cancel_button); cancelButton.setOnClickListener(new View.OnClickListener() { public void onClick(View view) { setResult(RESULT_CANCELED); finish(); } });
setResult函数的参数分别是:result code和result intent,也可以只包含result code。如果sub activity没执行setResult就返回(例如用户按”返回“键或在setResult之前调用了finish方法),那么result code将被设为RESULT_CANCELED,result intent为null。
sub Activity结束后,calling Activity中的回调方法onActivityResult将被调用,onActivityResult有这几个参数:
- Request Code:startActivityForResult方法的参数,用于确认被的调用sub Activity。
- Result Code: setResult参数一,任意整型值,一般是Activity.RESULT_OK或Activity.RESULT_CANCELED。
- Data:setResult中的参数二,Intent。
onActivityResult的示例代码:
private static final int SELECT_HORSE = 1; private static final int SELECT_GUN = 2; Uri selectedHorse = null; Uri selectedGun = null;
@Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); switch(requestCode) { case (SELECT_HORSE): if (resultCode == Activity.RESULT_OK) selectedHorse = data.getData(); break; case (SELECT_GUN): if (resultCode == Activity.RESULT_OK) selectedGun = data.getData(); break; default: break; } }
Android原生Action
- ACTION_ALL_APPS:显示设备中所有的APP,一般由launcher处理。
- ACTION_ANSWER:接听来电,一般由native in-call screen处理。
- ACTION_BUG_REPORT:报告bug:native bug-reporting mechanism。
- ACTION_CALL:尽量用ACTION_DIAL,ACTION_CALL会马上拨出电话。
- ACTION_CALL_BUTTON:响应用户按“接听”的硬件按钮事件。
- ACTION_DELETE:启动一个Activity并删除Intent中URI所指的资源。
- ACTION_DIAL:启动拨号器并将intent中的号码填进去。
- ACTION_EDIT:请求一个可以编辑Intent中URI所指的资源的Activity。
- ACTION_INSERT:打开一个Activity,这个Activity可以向Intent中的URI所指向的Cursor插入数据。当作为sub Activity被调用时,它应该返回指向新插入数据的URI。
- ACTION_PICK:类似ACTION_INSERT,用pick操作替换insert操作。
- ACTION_SEARCH:启动一个特定的search Activity。
- ACTION_SEARCH_LONG_PRESS:响应用户在设备上长按search键的事件。
- ACTION_SENDTO:启动一个Activity,此Activity可以给Intent中URI指定的Contact发送数据。
- ACTION_SEND:启动一个Activity,此Activity可以发送Intent中指定的数据。
- ACTION_VIEW:请求用最恰当的方式显示Intent中的数据。
- ACTION_WEB_SEARCH:打开浏览器进行搜索。