Intent 介绍
Intent 是 Android 程序中各组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,还可以在不同组件间传递数据。Intent 一般被用于启动活动、启动服务、或是发送广播等场景。Intent 的用法大致分为两种,显式 intent 和隐式 intent。<br />
既然需要涉及到第二个 Activity,那我们就先来创建一个新的 Activity,命名为 SecondActivity.
- Step 1 新建 second_layout.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">
<Button
android:id="@+id/button_2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Button 2"/>
</LinearLayout>
- Step 2 新建活动 SecondActivity 继承自 Activity
public class SecondActivity extends Activity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.second_layout);
}
}
- Step 3 在 AndroidManifest.xml 中为 SecondActivity 注册
<activity android:name=".SecondActivity">
</activity>
显式 Intent
Intent 有多个构造函数的重载,其中一个是 Intent(Context packageContext,Class<?>cls),这个构造函数接受两个参数,第一个 Context 是提供一个启动活动的上下文,第二个参数 Class 是指定想要启动的目标活动。通过此函数可以构建出 Intent 的“意图”。Activity 类中提供了一个 startActivity() 方法,此方法用于启动活动,它接收一个 Intent 参数。<br />
修改 FirstActivity 中按钮的点击事件,代码如下:
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
startActivity(intent);
}
});
使用这种方式来启动活动,Intent 的意图很明显,所以称为显式 Intent。
隐式 Intent
并不明确指出我们想要启动哪一个活动,而是指定了一系列更抽象的 action 和 category 等,然后又系统去分析这个 Intent 并帮我们找到合适的活动去启动。
- 配置 AndroidManifest.xml 中 <activity> 标签下 <intent-filter> 中的 <action> 和 <category> 可以指定当前活动能够相应的 action 和 category。
<activity android:name=".SecondActivity">
<intent-filter>
<action android:name="com.example.activitytest.ACTION_START"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
- 修改 FirstActivity 中按钮的点击事件,使 SecondActivity 能够响应。
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent("com.example.activitytest.ACTION_START");
startActivity(intent);
}
});
为什么这里没有指定 category 呢?因为 android.intent.category.DEFAULT 是一种默认的 category,startActivity() 方法会自动把 category 添加到 Intent 中。<br />
每个 Intent 中只能指定一个 action,但却能够指定多个 category。
- 再为 SecondActivity 添加一个 category。
<activity android:name=".SecondActivity">
<intent-filter>
<action android:name="com.example.activitytest.ACTION_START"/>
<category android:name="android.intent.category.DEFAULT"/>
<category android:name="com.example,activitytest.MY_CATEGORY"/>
</intent-filter>
</activity>
- 再次修改 FirstActivity 中按钮的点击事件,使 SecondActivity 能够响应。
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent("com.example.activitytest.ACTION_START");
intent.addCategory("com.example,activitytest.MY_CATEGORY");
startActivity(intent);
}
});
调用 Intent 中的 addCategory() 方法来添加一个 category,这样 SecondActivity 就能够正常响应了。
更多的隐式 Intent 用法
- 修改 FirstActivity 中按钮的点击事件,使之能够通过系统自带的浏览器打开网页。
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.baidu.com"));
startActivity(intent);
}
});
Intent 的 action 是一个 Android 系统内置的动作,常量值为 android.intent.action.VIEW。然后通过 Uri.parse() 方法,将一个网址字符串解析成一个 Uri 对象,再调用 Intent 的 setData() 方法把 Uri 对象传递进去。setDate() 方法接收一个 Uri 对象,主要用于指定当前 Intent 正在操作的数据,数据通常是以字符串形式传递到 Uri.parse() 方法中解析产生。
<data> 标签的使用
在 <intent-filter> 标签中再配置一个 <data> 标签,可用于更精准的指定当前活动能够响应什么类型的数据。<data> 标签主要可以配置以下内容:
- android:scheme <br />
用于指定数据协议部分 - android:host <br />
用于指定数据主机名部分 - android:port <br />
用于指定数据端口部分 - android:path <br />
用于指定主机名和端口之后部分 - android:mimeType <br />
用于指定可以处理的数据类型
只有 <data> 标签中指定的内容和 Intent 中携带的 Data 完全一致时,当前活动才能够响应该 Intent。例如:
- 新建 third_layout.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">
<Button
android:id="@+id/button_3"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Button 3"/>
</LinearLayout>
- 新建活动 ThirdActivity 继承自 Activity
public class ThirdActivity extends Activity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.third_layout);
}
}
- 在 AndroidManifest.xml 中为 ThirdActivity 注册
<activity android:name=".ThirdActivity">
<intent-filter>
<action android:name="android.intent.action.VIEW"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:scheme="http"/>
</intent-filter>
</activity>
此时运行点击 FirstActivity 的按钮会弹出一个列表,让你选择是用自带浏览器打开还是使用新建的 ThirdActivity 打开。
调用系统拨号界面
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(Intent.ACTION_DIAL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
}
});
向下一个活动传递数据
启动活动时传递数据思路很简单,Intent 中提供了一系列的 putExtra() 方法的重载,把想要传递的数据暂存在 Intent 中,启动另一个活动后,再把数据从 Intent 中取出来。
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
String data = "Hello SecondActivity";
Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
intent.putExtra("extra_data",data);
startActivity(intent);
}
});
putExtra() 方法接收两个参数,第一个是键,第二个是值。
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.second_layout);
Intent intent = getIntent();
String data = intent.getStringExtra("extra_data");
Log.d("SecondActivity",data);
}
在 SecondActivity 中将传递的数据取出,并打印出来。通过 getIntent() 方法获取用于启动 SecondActivity 的 Intent ,然后调用 getStringExtra() 方法取出数据。
返回数据给上一活动
Activity 中有一个 startActivityForResult() 方法也用于启动活动,但这方法期望在活动销毁时返回一个结果给上一个活动。
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
startActivityForResult(intent,1);
}
});
使用 startActivityForResult() 方法来启动 SecondActivity ,startActivityForResult() 方法接收两个参数,第一个参数是 Intent,第二个参数是请求码,用于在回调中判断数据来源。
- 为 SecondActivity 按钮注册点击事件,添加返回数据逻辑
button2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent();
intent.putExtra("data_return","Hello FirstActivity");
setResult(RESULT_OK,intent);
finish();
}
});
setResult() 方法专门用于向上一个活动返回数据,接收两个参数,第一个是向上一个活动返回处理结果,第二个参数是把带有数据的 Intent 传递回去,然后调用 finish() 方法销毁当前活动。<br />
使用 startActivityForResult() 方法来启动 SecondActivity,在 SecondActivity 销毁之后会回调上一个活动的 onActivityResult() 方法,所以需在 FristActivity 上重写这个方法来获取数据。
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
switch (requestCode){
case 1:
if (resultCode == RESULT_OK){
String returnedData = data.getStringExtra("data_return");
Log.d("FirstActivity",returnedData);
}
break;
default:
}
}
onActivityResult() 方法带有三个参数,第一个是启动活动时传入的请求码,第二个是返回时的处理结果,第三个是携带返回数据的 Intent。<br /><br />
如果用户点击的是 Back 键,而不是点击按钮怎么办?重写 onBackPressed() 方法。
@Override
public void onBackPressed() {
Intent intent = new Intent();
intent.putExtra("data_return","Hello FirstActivity");
setResult(RESULT_OK,intent);
finish();
}