Intent应用详解
安卓中的组件之间的联系,全部通过Intent,下面介绍Intent在四大组件之间如何使用:
启动一个Activity:Context.startActivity(Intent intent);
启动一个Service:Context.startService(Intent service);
绑定一个Service:Context.bindService(Intent service, ServiceConnection conn, int flags);
发送一个Broadcast:Context.sendBroadcast(Intent intent);
简单来说,Intent是系统各组件之间进行数据传递的数据负载者。当我们需要做一个调用动作,我们就可以通过Intent告诉Android系统来完成这个过程,Intent就是调用通知的一种操作。
Intent的六个核心属性包括:Component Name:组件名称;Action:动作;Data:数据;Category:分类;Extra:附加信息;Flag:标志。下面我们将会逐一介绍:
1.action,要执行的动作
对于有如下声明的Activity:
TargetActivity在其
public void gotoTargetActivity(View view) {
Intent intent = new Intent("com.scott.intent.action.TARGET");
startActivity(intent);
}
当我们为Intent指定相应的action,然后调用startActivity方法后,系统会根据action跳转到对应的Activity。
2.data和extras,即执行动作要操作的数据和传递到目标的附加信息
下面就举一个与浏览器交互的例子:
/**
* 打开指定网页
* @param view
*/
public void invokeWebBrowser(View view) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.google.com.hk"));
startActivity(intent);
}
/**
* 进行关键字搜索
* @param view
*/
public void invokeWebSearch(View view) {
Intent intent = new Intent(Intent.ACTION_WEB_SEARCH);
intent.putExtra(SearchManager.QUERY, "android"); //关键字
startActivity(intent);
}
上面两个方法分别是启动浏览器并打开指定网页、进行关键字搜索,分别对应的action是Intent.ACTION_VIEW和Intent.ACTION_WEB_SEARCH,前者需指定相应的网页地址,后者需指定关键字信息,对于关键字搜索来说,浏览器会按照自己设置的默认的搜索引擎进行搜索。
我们注意到,在打开网页时,为Intent指定一个data属性,这其实是指定要操作的数据,是一个URI的形式,我们可以将一个指定前缀的字符串转换成特定的URI类型,如:“http:”或“https:”表示网络地址类型,“tel:”表示电话号码类型,“mailto:”表示邮件地址类型,等等。例如,我们要呼叫给定的号码,可以这样做:
public void call(View view) {
Intent intent = new Intent(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:12345678"));
startActivity(intent);
}
那么我们如何知道目标是否接受这种前缀呢?这就需要看一下目标中元素的匹配规则了。
在目标标签中包含了以下几种子元素,他们定义了url的匹配规则:
android:scheme 匹配url中的前缀,除了“http”、“https”、“tel”...之外,我们可以定义自己的前缀
android:host 匹配url中的主机名部分,如“google.com”,如果定义为“*”则表示任意主机名
android:port 匹配url中的端口
android:path 匹配url中的路径
我们改动一下TargetActivity的声明信息:
这个时候如果只指定action就不够了,我们需要为其设置data值,如下:
public void gotoTargetActivity(View view) {
Intent intent = new Intent("com.scott.intent.action.TARGET");
intent.setData(Uri.parse("scott://com.scott.intent.data:7788/target"));
startActivity(intent);
}
此时,url中的每个部分和TargetActivity配置信息中全部一致才能跳转成功,否则就被系统拒绝。
putExtra方法,将关键字做为参数放置在Intent中,我们成为extras(附加信息),这里面涉及到了一个Bundle对象。
Bundle和Intent有着密不可分的关系,主要负责为Intent保存附加参数信息,它实现了android.os.Paracelable接口,内部维护一个Map类型的属性,用于以键值对的形式存放附加参数信息。在我们使用Intent的putExtra方法放置附加信息时,该方法会检查默认的Bundle实例为不为空,如果为空,则新创建一个Bundle实例,然后将具体的参数信息放置到Bundle实例中。我们也可以自己创建Bundle对象,然后为Intent指定这个Bundle即可,如下:
public void gotoTargetActivity(View view) {
Intent intent = new Intent("com.scott.intent.action.TARGET");
Bundle bundle = new Bundle();
bundle.putInt("id", 0);
bundle.putString("name", "scott");
intent.putExtras(bundle);
startActivity(intent);
}
需要注意的是,在使用putExtras方法设置Bundle对象之后,系统进行的不是引用操作,而是复制操作,所以如果设置完之后再更改bundle实例中的数据,将不会影响Intent内部的附加信息。那我们如何获取设置在Intent中的附加信息呢?与之对应的是,我们要从Intent中获取到Bundle实例,然后再从中取出对应的键值信息:
Bundle bundle = intent.getExtras();
int id = bundle.getInt("id");
String name = bundle.getString("name");
当然我们也可以使用Intent的getIntExtra和getStringExtra方法获取,其数据源都是Intent中的Bundle类型的实例对象。
例如:在我们的应用主界面Activity通常有如下配置
代表该目标Activity是该应用所在task中的初始Activity并且出现在系统launcher的应用列表中。
几个常见的category如下:
Intent.CATEGORY_DEFAULT(android.intent.category.DEFAULT) 默认的category
Intent.CATEGORY_PREFERENCE(android.intent.category.PREFERENCE) 表示该目标Activity是一个首选项界面;
Intent.CATEGORY_BROWSABLE(android.intent.category.BROWSABLE)指定了此category后,在网页上点击图片或链接时,系统会考虑将此目标Activity列入可选列表,供用户选择以打开图片或链接。
在为Intent设置category时,应使用addCategory(String category)方法向Intent中添加指定的类别信息,来匹配声明了此类别的目标Activity。
例如:一个可以处理图片的目标Activity在其声明中包含这样的mimeType:
5.component,目标组件的包或类名称
intent.setComponent(new ComponentName(getApplicationContext(), TargetActivity.class));
intent.setComponent(new ComponentName(getApplicationContext(), "com.scott.intent.TargetActivity"));
intent.setComponent(new ComponentName("com.scott.other", "com.scott.other.TargetActivity"));
其中,前两种是用于匹配同一包内的目标,第三种是用于匹配其他包内的目标。需要注意的是,如果我们在Intent中指定了component属性,系统将不会再对action、data/type、category进行匹配。
接下来是分别是隐式意图和显示意图打开一个activity的例子,该例子分别是发送短信的例子,(1)隐式意图是短信内容直接跳到系统的发短信的界面,(原始页面A,发短信页面(系统页面)B,将页面A的数据通过Intent 传递给页面B)(2)显示意图是自己制作联系人列表,短信内容模板,在发短信界面分别打开两个页面后数据回调到短信页面,(原始页面A 要开启页面B ,打开B页面之后,选中条目,关闭界面B,将B页面的数据回调给页面A):
隐式意图:
UI:
Mainactivity代码:
package com.example.sendMessage2;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;
public class MainActivity extends Activity implements OnItemClickListener{
//ListView
private ListView messagelist;
//建立数据源
String meStrings [] = {"我在开会,请稍后联系","我在吃饭,请稍后联系",
"我在打代码,请稍后联系","我在开车,请稍后联系","我在约会,请稍后联系"};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
messagelist = (ListView) findViewById(R.id.listView1);
//建立适配器
ArrayAdapter adapter = new ArrayAdapter(this, android.R.layout.simple_list_item_1, meStrings);
//设置适配器
messagelist.setAdapter(adapter);
//设置子条目点击事件
messagelist.setOnItemClickListener(this);
}
@Override
public void onItemClick(AdapterView> parent, View view, int position, long id) {
String string = meStrings[position];
Intent intent = new Intent();
//intent.setAction("");
intent.setAction("android.intent.action.SEND");
//intent.addCategory(string);
intent.addCategory("android.intent.category.DEFAULT");
//intent.setType(string);
intent.setType("text/plain");
//发送数据
//intent.putExtra("", string);
intent.putExtra("sms_body", string); //要求大家知道这个是怎么来的 ?? 查看源码 ,这个“sms_body”是短信中的源码
startActivity(intent);
}
}
显示意图:
UI这里只提供mainactivity界面:
Mainactivity代码:
package com.example.sendMesssage;
/*
* 使用显示意图进行短信发送
*
*/
import java.util.ArrayList;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.telephony.SmsManager;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
public class MainActivity extends Activity {
//联系人
private String phone;
//短信内容
private String message;
private EditText adress,content;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//联系人
adress = (EditText) findViewById(R.id.et_number);
//短信内容
content = (EditText) findViewById(R.id.et_sms_content);
}
//添加联系人
public void add(View view){
//跳转到联系人界面
Intent intent = new Intent(this,contact.class);
startActivityForResult(intent, 1);
}
//添加短信内容
public void insert(View view){
//跳转到短信内容界面
Intent intent = new Intent(this,MessageActivity.class);
startActivityForResult(intent, 2);
}
//发送短信
public void click(View view){
//(2)实现发送短信的功能 smsManager 获取这个类的实例
SmsManager smsManager = SmsManager.getDefault();
//(3)分割短信 分条发送
ArrayList divideMessages = smsManager.divideMessage(message);
for (String div : divideMessages) {
//(4)发送短信 s
smsManager.sendTextMessage(phone, null, div, null, null);
}
Toast.makeText(this, "发送完毕", 0).show();;
}
//数据回调
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
// TODO Auto-generated method stub
super.onActivityResult(requestCode, resultCode, data);
if(1 == requestCode){
String phone = data.getStringExtra("phone");
adress.setText(phone);
}
if(2 == requestCode){
String message = data.getStringExtra("messages");
content.setText(message);
}
}
}
contact代码:
package com.example.sendMesssage;
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
import android.content.Intent;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;
public class contact extends Activity implements OnItemClickListener{
//建立数据源
ArrayList phones;
//建立适配器
ArrayAdapter adapter;
//建立一个ListView
ListView listView;
protected void onCreate(android.os.Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.contact_item);
//实例化一个ListView
listView = (ListView) findViewById(R.id.phonelist);
phones = new ArrayList();
//建立数据源
for(int i = 0 ;i<= 5; i++){
phones.add("188234"+i);
}
//建立适配器
adapter = new ArrayAdapter(this,android.R.layout.simple_list_item_1, phones);
//设置适配器
listView.setAdapter(adapter);
listView.setOnItemClickListener(this);
}
@Override
public void onItemClick(AdapterView> parent, View view, int position, long id) {
// TODO Auto-generated method stub
//获取数据
String phone = phones.get(position);
//传递数据
Intent intent = new Intent();
intent.putExtra("phone", phone);
setResult(11, intent);
finish();
};
}
package com.example.sendMesssage;
import android.app.Activity;
import android.content.Intent;
import android.nfc.Tag;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
/*
* 该页面是短信模板界面
*/
public class MessageActivity extends Activity implements OnItemClickListener{
private ListView messagelist;
//建立数据源
String meStrings [] = {"我在开会,请稍后联系","我在吃饭,请稍后联系",
"我在打代码,请稍后联系","我在开车,请稍后联系","我在约会,请稍后联系"};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.message_item);
messagelist = (ListView) findViewById(R.id.messagelist);
//建立适配器
ArrayAdapter adapter = new ArrayAdapter(this, android.R.layout.simple_list_item_1, meStrings);
//设置适配器
messagelist.setAdapter(adapter);
messagelist.setOnItemClickListener(this);
}
@Override
public void onItemClick(AdapterView> parent, View view, int position, long id) {
Log.i("tag", "到我了");
//获取短信内容
String s = meStrings[position];
//通过Intent传递给上一个界面
Intent intent = new Intent();
intent.putExtra("messages", s);
setResult(12, intent);
//关闭此界面
finish();
}
}