《第一行代码》学习笔记:Intent--在活动之间穿梭

Intent是Android程序中各组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,还可以在不同组件之间传递数据。Intent一般可被用于启动活动、启动服务以及发送广播等场景。

启动活动
创建一个activity,自动生成布局文件。此时AS会在AndroidManifest.xml自动注册这个活动,如果不是主活动,也不需要配置<intent-filter> 标签里的内容

一、显示使用intent(一步到位,显示使用)
Intent有多个构造函数的重载,其中一个是Intent(Context packageContext, Class<?>cls) 。这个构造函数接收两个参数,第一个参数Context 要求提供一个启动活动的上下文,第二个参数Class 则是指定想要启动的目标活动通过这个构造函数就可以构建出Intent的“意图”。然后我们应该怎么使用这个Intent呢?Activity类中提供了一个startActivity() 方法,这个方法是专门用于启动活动的,它接收一个Intent 参数,这里我们将构建好的Intent传入startActivity() 方法就可以启动目标活动了。
使用示例:修改主活动中的按钮点击事件,构建出Intent

button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
startActivity(intent);
}
});

意图是在FirstActivity这个活动的基础上打开SecondActivity这个活动。然后通过startActivity() 方法来执行这个Intent。

二、隐式使用intent(需要两步,配置activity+响应)
不明确指出我们想要启动哪一个活动,而是指定了一系列更为抽象的action 和category 等信息,然后交由系统去分析这个Intent,并帮我们找出合适的活动去启动。

通过在<activity> 标签下配置<intent-filter> 的内容,可以指定当前活动能够响应的
action 和category ,使用时,打开AndroidManifest.xml,添加如下代码:

<activity android:name=".SecondActivity" >
<intent-filter>
<action android:name="com.example.activitytest.ACTION_START" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>

<action> 标签中我们指明了当前活动可以响com.example.activitytest.ACTION_START 这个action ,而<category> 标签则包含了一些附加信息,更精确地指明了当前的活动能够响应的Intent中还可能带有的category 。只有<action><category> 中的内容同时能够匹配上Intent中指定的action 和category时,这个活动才能响应该Intent。修改(一)中FirstActivity中按钮的点击事件,代码如下所示:

button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.example.activitytest.ACTION_START");
startActivity(intent);
}
});

使用了Intent的另一个构造函数,直接将action 的字符串传了进去,表明我们想要启动能够响应com.example.activitytest.ACTION_START 这个action 的活动。那前面不是说要<action> 和<category> 同时匹配上才能响应的吗?怎么没看到哪里有指定category 呢?这是因为android.intent.category.DEFAULT 是一种默认的category,在调用startActivity() 方法的时候会自动将这个category 添加到Intent中。

每个Intent中只能指定一个action ,但却能指定多个category 。目前我们的Intent中只有一个默认的category,添加方法intent.addCategory("com.example.activitytest.MY_CATEGORY");
注意要在对应活动的<intent-filter> 标签中声明可以响应这个category 。

三、隐式Intent启动其他程序的活动
修改FirstActivity中按钮点击事件的代码(一步到位),如下所示:

button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.baidu.com"));
startActivity(intent);
}
});

指定了Intent的action 是Intent.ACTION_VIEW ,这是一个Android系统内置的动作,其常量值为android.intent.action.VIEW 。然后通过Uri.parse() 方法,将一个网址字符串解析成一个Uri 对象,再调用Intent的setData() 方法将这个Uri 对象传递进去。
setdata接收一个Uri 对象,主要用于指定当前Intent正在操作的数据,而这些数据通
常都是以字符串的形式传入到Uri.parse() 方法中解析产生的。

<intent-filter> 标签中再配置一个<data> 标签,用于更精确地
指定当前活动能够响应什么类型的数据。<data> 标签中主要可以配置以下内容。
android:scheme 。用于指定数据的协议部分,如上例中的http部分。
android:host 。用于指定数据的主机名部分,如上例中的www.baidu.com部分。
android:port 。用于指定数据的端口部分,一般紧随在主机名之后。
android:path 。用于指定主机名和端口之后的部分,如一段网址中跟在域名之后的内
容。
android:mimeType 。用于指定可以处理的数据类型,允许使用通配符的方式进行指
定。
只有<data> 标签中指定的内容和Intent中携带的Data完全一致时,当前活动才能够响该
Intent。不过一般在<data> 标签中都不会指定过多的内容,如上面浏览器示例中,其实需要指定android:scheme 为http,就可以响应所有的http协议的Intent了。

除了http协议外,我们还可以指定很多其他协议,比如geo表示显示地理位置、tel表示拨打电话。下面的代码展示了如何在我们的程序中调用系统拨号界面。

button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(Intent.ACTION_DIAL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
}
});

四、Intent在启动活动的时候传递数据(向下一个活动传递数据)
启动活动时传递数据的思路很简单,Intent中提供了一系列putExtra() 方法的重载,可以把我们想要传递的数据暂存在Intent中,启动了另一个活动后,只需要把这些数据再从Intent中取出就可以了。比如说FirstActivity中有一个字符串,现在想把这个字符串传递到SecondActivity中,
你就可以这样编写:

button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String data = "Hello SecondActivity";
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
intent.putExtra("extra_data", data);
startActivity(intent);
}
});

这里我们还是使用显式Intent的方式来启动SecondActivity,并通过putExtra() 方法传递了一个字符串。注意这里putExtra() 方法接收两个参数,第一个参数是键,用于后面从Intent中取值,第二个参数才是真正要传递的数据。

然后我们在SecondActivity中将传递的数据取出,并打印出来,代码如下所示:

public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.second_layout);
Intent intent = getIntent();
String data = intent.getStringExtra("extra_data");
Log.d("SecondActivity", data);
}
}

首先可以通过getIntent() 方法获取到用于启动SecondActivity的Intent,然后调用getStringExtra() 方法,传入相应的键值,就可以得到传递的数据了。这里由于我们传
递的是字符串,所以使用getStringExtra() 方法来获取传递的数据。如果传递的是整型数
据,则使用getIntExtra() 方法;如果传递的是布尔型数据,则使用getBooleanExtra()
方法,以此类推。
重新运行程序,在FirstActivity的界面点击一下按钮会跳转到SecondActivity,查看logcat打印信息,可以看到,我们在SecondActivity中成功得到了从FirstActivity传递过来的数据

五、intent返回数据给上一个活动
(1)在上一个活动中的点击事件中使用startActivityForResult() 方法
(2)
Activity中还有一个startActivityForResult() 方法也是用于启动活动的,但这个方法期望在活动销毁的时候能够返回一个结果给上一个活动。毫无疑问,这就是我们所需要的。

startActivityForResult() 方法接收两个参数,第一个参数还是Intent,第二个参数是请
求码,用于在之后的回调中判断数据的来源。我们还是来实战一下,修改FirstActivity中按钮的
点击事件,代码如下所示:

button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
startActivityForResult(intent, 1);
}
});

这里我们使用了startActivityForResult() 方法来启动SecondActivity,请求码只要是一
个唯一值就可以了,这里传入了1。接下来我们在SecondActivity中给按钮注册点击事件,并在点
击事件中添加返回数据的逻辑,代码如下所示:

public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.second_layout);
Button button2 = (Button) findViewById(R.id.button_2);
button2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent();
intent.putExtra("data_return", "Hello FirstActivity");
setResult(RESULT_OK, intent);
finish();
}
});
}
}

可以看到,我们还是构建了一个Intent,只不过这个Intent仅仅是用于传递数据而已,它没有指定任何的“意图”。紧接着把要传递的数据存放在Intent中,然后调用了setResult() 方法。这个方法非常重要,是专门用于向上一个活动返回数据的。setResult() 方法接收两个参数,第一个参数用于向上一个活动返回处理结果,一般只使用RESULT_OK 或RESULT_CANCELED 这两个值,第二个参数则把带有数据的Intent传递回去,然后调用了finish() 方法来销毁当前活动。

由于我们是使用startActivityForResult() 方法来启动SecondActivity的,在SecondActivity被销毁之后会回调上一个活动的onActivityResult() 方法,因此我们需要在FirstActivity中重写这个方法来得到返回的数据,如下所示:

@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() 方法带有三个参数,第一个参数requestCode ,即我们在启动活动
时传入的请求码。第二个参数resultCode ,即我们在返回数据时传入的处理结果。第三个参数data ,即携带着返回数据的Intent。由于在一个活动中有可能调
用startActivityForResult() 方法去启动很多不同的活动,每一个活动返回的数据都会回
调到onActivityResult() 这个方法中,因此我们首先要做的就是通过检查requestCode
的值来判断数据来源。确定数据是从SecondActivity返回的之后,我们再通过resultCode 的值来判断处理结果是否成功。最后从data 中取值并打印出来,这样就完成了向上一个活动返回数据的工作。重新运行程序,在FirstActivity的界面点击按钮会打开SecondActivity,然后在SecondActivity界面点击Button 2按钮会回到FirstActivity,这时查看logcat的打印信息。

如果用户在SecondActivity中并不是通过点击按钮,而是通过按下Back键回到FirstActivity在,那么怎么返回数据?SecondActivity中重写onBackPressed() 方法来解决这个问题,代码如下所示:

@Override
public void onBackPressed() {
Intent intent = new Intent();
intent.putExtra("data_return", "Hello FirstActivity");
setResult(RESULT_OK, intent);
finish();
}

这样的话,当用户按下Back键,就会去执行onBackPressed() 方法中的代码,我们在这里添加返回数据的逻辑就行了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值