Android读书笔记之使用Intent和IntentFilter进行通信

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)));
	}
}


在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值