Android读书笔记之使用Intent和IntentFilter进行通信
背景:
Andorid使用Intent来封装程序的调用意图,不管程序想启动一个Activity也好,还是一个Service组件也好,或者BroadcastReceiver也好,Android使用统一的Intent对象来封装这种启动意图,很明显使用Intent提供了一致的编程模型。
Intent还是应用程序组件之间通信的重要媒介,正如从前面程序中所看到的,两个Activity可以把需要交换的数据封装成Bundle对象,然后使用Intent来携带Bundle对象,这样就实现了两个Activity之间的数据交换。
一.Intent对象简述
Android的应用程序包含三种重要的组件:Activity,Service,BroadcastReceiver,应用程序采用了一致的方式来启动他们----都是依靠Intent来启动的,Intent就封装了程序想要启动程序的意图。不仅如此,Intent还可用于与被启动组件交换信息。
Intent对象大致包含Component,Action,Category,Data, Type, Extra和Flag这7种属性,其中Component用于明确指定需要启动的目标组件,而Extra则用于携带需要交换的数据的。
二.Intent的属性及intent-filter配置
1.Component属性
Intent的Component属性需要接受一个ComponentName对象,ComponentName对象包含构造器。
Android应用的Context代表了访问该应用环境信息的接口,而Android应用的包名则作为应用的唯一标识,因此Android应用的Context对象与该应用的包名有一一对应的关系。上面三个setClass()方法正是指定组件的包名和实现类。
指定Component属性的Intent已经明确了它将要启动哪个组件,因此这种Intent也被称为显示Intent。没有指定Component属性的Intent被称为隐式Intent。
MainActivity.java
public class MainActivity extends Activity
{
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button bn = (Button) findViewById(R.id.bn);
// 为bn按钮绑定事件监听器
bn.setOnClickListener(new OnClickListener()
{
@Override
public void onClick(View arg0)
{
// 创建一个ComponentName对象
ComponentName comp = new ComponentName(MainActivity.this,
SecondActivity.class);
Intent intent = new Intent();
// 为Intent设置Component属性
intent.setComponent(comp);
startActivity(intent);
}
});
}
}
上面代码创建ComponentName对象,并将该对象设置成Intent对象的Component属性,这样应用程序即可根据该Intent的意图去启动指定组件。
上面可以简化为:
//根据指定组件类来创建Intent
Intent intent = new Intent(ComponentAttr.this, SecondActivity.class);
当程序通过Intent的Component属性(明确指定了启动哪个组件)启动特定组件时,被启动组件几乎不需要使用<intent-filter…/>元素进行配置。
SecondActivity.java
public class SecondActivity extends Activity
{
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.second);
EditText show = (EditText) findViewById(R.id.show);
// 获取该Activity对应的Intent的Component属性
ComponentName comp = getIntent().getComponent();
// 显示该ComponentName对象的包名、类名
show.setText("组件包名为:" + comp.getPackageName()
+ "\n组件类名为:" + comp.getClassName());
}
}
2.Action Category属性与intent-filter配置
Intent的Action,Category属性的值都是一个普通的字符串,其中Action代表该Intent所要完成的一个抽象动作,而Category则用于为Action增加额外的附加类别信息。通常Action属性会与Category属性结合使用。
下面通过一个简单的示例来示范Action属性的作用。下面第一个Activity非常简单,它只包括一个普通按钮,当用户单击该按钮时,程序会“跳转”到第二个Activity,但第一个Activity指定跳转的Intent时,并不以“硬编码”的方式指定要跳转的目标Activity,而是为Intent指定Action属性。
public class MainActivity extends Activity
{
public final static String CRAZYIT_ACTION =
"org.crazyit.intent.action.CRAZYIT_ACTION";
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button bn = (Button) findViewById(R.id.bn);
// 为bn按钮绑定事件监听器
bn.setOnClickListener(new OnClickListener()
{
@Override
public void onClick(View arg0)
{
// 创建Intent对象
Intent intent = new Intent();
// 为Intent设置Action属性(属性值就是一个普通字符串)
intent.setAction(MainActivity.CRAZYIT_ACTION);
startActivity(intent);
}
});
}
}
这里是根据Intent来启动Activity----但该Intent并未以“硬编码”的方式指定要启动哪个Activity,这取决于Activity配置中<intent-filter…/>元素的配置。
<intent-filter…/>元素是AndroidManifest.xml文件中<activity…/>元素的子元素,通常可包含如下子元素:
0-N个<action…/>子元素;
0-N个<category…/>子元素;
0-1个<data…/>子元素。
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="org.crazyit.intent" >
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name=".SecondActivity"
android:label="@string/app_name">
<intent-filter>
<!-- 指定该Activity能响应Action为指定字符串的Intent -->
<action android:name="org.crazyit.intent.action.CRAZYIT_ACTION" />
<!-- 指定该Activity能响应Action属性为helloWorld的Intent,这行只是实验,表明
Activity能响应Action属性值为helloworld字符串,
Category属性值为android.intent.category.DEFAULT的Intent -->
<action android:name="helloWorld" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
</application>
</manifest>
SecondActivity.java
public class SecondActivity extends Activity
{
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.second);
EditText show = (EditText) findViewById(R.id.show);
// 获取该Activity对应的Intent的Action属性
String action = getIntent().getAction();
// 显示Action属性
show.setText("Action为:" + action);
}
}
3.指定Action,Category调用系统Activity
实际上Intent对象不仅可以启动本应用内程序组件,也可以启动android系统的其他应用的程序组件,包括系统自带的程序组件-----只要权限允许。
实例: 查看并获取联系人电话
该实例将会在程序中提供一个按钮,用户单击该按钮时会显示系统的联系人列表,当用户单击指定联系人之后,程序将会显示该联系人的名字,电话。
main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_horizontal">
<!-- 显示联系人姓名的文本框 -->
<EditText
android:id="@+id/show"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:editable="false"
android:cursorVisible="false"/>
<!-- 显示联系人的电话的文本框 -->
<EditText
android:id="@+id/phone"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:editable="false"
android:cursorVisible="false"/>
<Button
android:id="@+id/bn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="查看联系人"/>
</LinearLayout>
MainActivity.java
public class MainActivity extends Activity
{
final int PICK_CONTACT = 0;
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button bn = (Button) findViewById(R.id.bn);
// 为bn按钮绑定事件监听器
bn.setOnClickListener(new OnClickListener()
{
@Override
public void onClick(View arg0)
{
// 创建Intent
Intent intent = new Intent();
// 设置Intent的Action属性
intent.setAction(Intent.ACTION_GET_CONTENT);
// 设置Intent的Type属性
intent.setType("vnd.android.cursor.item/phone");
// 启动Activity,并希望获取该Activity的结果
startActivityForResult(intent, PICK_CONTACT);
}
});
}
@Override
public void onActivityResult(int requestCode
, int resultCode, Intent data)
{
super.onActivityResult(requestCode, resultCode, data);
switch (requestCode)
{
case (PICK_CONTACT):
if (resultCode == Activity.RESULT_OK)
{
// 获取返回的数据
Uri contactData = data.getData();
CursorLoader cursorLoader = new CursorLoader(this
, contactData, null, null, null, null);
// 查询联系人信息
Cursor cursor = cursorLoader.loadInBackground();
// 如果查询到指定的联系人
if (cursor.moveToFirst())
{
String contactId = cursor.getString(cursor
.getColumnIndex(
ContactsContract.Contacts._ID));
// 获取联系人的名字
String name = cursor.getString(cursor
.getColumnIndexOrThrow(
ContactsContract.Contacts.DISPLAY_NAME));
String phoneNumber = "此联系人暂未输入电话号码";
//根据联系人查询该联系人的详细信息
Cursor phones = getContentResolver().query(
ContactsContract.CommonDataKinds.
Phone.CONTENT_URI, null,
ContactsContract.CommonDataKinds.Phone.CONTACT_ID
+ " = " + contactId, null, null);
if (phones.moveToFirst())
{
//取出电话号码
phoneNumber = phones
.getString(phones
.getColumnIndex(ContactsContract
.CommonDataKinds.Phone.NUMBER));
}
// 关闭游标
phones.close();
EditText show =(EditText)findViewById(R.id.show);
//显示联系人的名称
show.setText(name);
EditText phone =(EditText)findViewById(R.id.phone);
//显示联系人的电话号码
phone.setText(phoneNumber);
}
// 关闭游标
cursor.close();
}
break;
}
}
}
还要在AndroidManifest.xml文件中增加如下配置。
<uses-permission android:name="android.permission.READ_CONTACTS" />
另外返回系统Home桌面可以使用如下Intent:
满足如下该Intent的Activity其实就是Android系统的Home桌面了的。
//创建Intent对象
Intent intent = new Intent();
//为Intent设置Action,Category属性
intent.setAction(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_HOME);
startActivity(intent);
4.Data,Type属性与intent-filter配置
Data属性通常用于向Action属性提供操作的数据。Data属性接受一个Uri对象,一个Uri对象通常通过如下形式的字符串来表示:
content://com.android.contacts/contacts/1
tel:123
Uri字符串总满足如下格式:
scheme://host:port/path
Type属性用于指定该Data属性所指定Uri对应的MIME类型,
Data属性与Type属性的关系比较微妙,这两个属性会相互覆盖。
1> 如果为Intent先设置Data属性,后设置Type属性,那么Type属性将会覆盖Data属性;
2> 如果为Intent先设置Type属性,后设置Data属性,那么Data属性将会覆盖Type属性;
3> 如果希望Intent既有Data属性,也有Type属性,则应该调用Intent的setDataAndType()方法。
下面示例演示了Intent的Data属性与Type属性互相覆盖的情形,该示例的界面布局文件很简单,只是定义了三个按钮,并为三个按钮绑定了事件处理函数。
MainActivity.java
public class MainActivity extends Activity
{
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
}
public void overrideType(View source)
{
Intent intent = new Intent();
// 先为Intent设置Type属性
intent.setType("abc/xyz");
// 再为Intent设置Data属性,覆盖Type属性
intent.setData(Uri.parse("lee://www.fkjava.org:8888/test"));
Toast.makeText(this, intent.toString(), Toast.LENGTH_LONG).show();
}
public void overrideData(View source)
{
Intent intent = new Intent();
// 先为Intent设置Data属性
intent.setData(Uri.parse("lee://www.fkjava.org:8888/mypath"));
// 再为Intent设置Type属性,覆盖Data属性
intent.setType("abc/xyz");
Toast.makeText(this, intent.toString(), Toast.LENGTH_LONG).show();
}
//同时设置了Data, Type属性,这样该Intent中才会同时具有Data,Type属性。
public void dataAndType(View source)
{
Intent intent = new Intent();
// 同时设置Intent的Data、Type属性
intent.setDataAndType(Uri.parse("lee://www.fkjava.org:8888/mypath"),
"abc/xyz");
Toast.makeText(this, intent.toString(), Toast.LENGTH_LONG).show();
}
}
在AndroidManifest.xml文件中为组件声明Data,Type属性都通过<data…/>元素
<data android:mineType=""
android:scheme=""
android:host=""
android:port=""
android:path=""
android:pathPrefix=""
android:pathPattern="" />
Intent的Type属性也用于指定该Intent的要求,对应组件中<intent-filter…/>元素的<data…/>子元素的mimeType属性必须与此相同,才能启动该组件。
Intent的Data属性则略有差异,程序员为Intent指定Data属性时,Data属性的Uri对象实际上可分为scheme,host,port和path部分,此时并不要求被启动组件的<intent-filter…/>中<data…/>子元素的android:scheme,android:host,android:port,android:path完全满足。
5.Extra属性
Intent的Extra属性通常用于在多个Action之间进行数据交换,Intent的Extra属性值应该是一个Bundle对象,Bundle对象就像一个Map对象,它可以存入多个key-value对,这样就可以通过Intent在不同Activity之间进行数据交换了的。
6.Flag属性
Intent的Flag属性用于为该Intent添加一些额外的控制旗标,Intent可调用addFlags()方法来添加控制旗标。
三.使用Intent创建Tab页
setContent(Intent intent):直接将指定Intent对应的Activity设置成Tab页的Content。
MainActivity.java
package org.crazyit.intent;
import android.app.Activity;
import android.app.TabActivity;
import android.content.Intent;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TabHost;
public class MainActivity extends TabActivity
{
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// 获取该Activity里面的TabHost组件
TabHost tabHost = getTabHost();
// 使用Intent添加第一个Tab页面
tabHost.addTab(tabHost
.newTabSpec("tab1")
.setIndicator("已接电话",
getResources().getDrawable(R.drawable.ic_launcher))
.setContent(new Intent(this, BeCalledActivity.class)));
// 使用Intent添加第二个Tab页面
tabHost.addTab(tabHost.newTabSpec("tab1")
.setIndicator("呼出电话")
.setContent(new Intent(this, CalledActivity.class)));
// 使用Intent添加第三个Tab页面
tabHost.addTab(tabHost.newTabSpec("tab1")
.setIndicator("未接电话")
.setContent(new Intent(this, NoCallActivity.class)));
}
}