intent 组件隐式调用_Android初恋第二季之二: 使用【Intent】在【活动】之间穿梭...

使用【Intent】在活动之间穿梭

只有一个活动的应用也太简单了吧?没错,你的追求应该更高一点。不管你想创建多少个活动,方法都和上一节中介绍的是一样的。唯一的问题在于,你在启动器中点击应用的图标只会进入到该应用的主活动,那么怎样才能由主活动跳转到其他活动呢?我们现在就来一起看一看。


使用显式 Intent

你应该已经对创建活动的流程比较熟悉了,那我们现在快速地在 ActivityTest 项目中再创建一个活动。

仍然还是右击 com.example.activitytest 包 →New→Activity→ Empty Activity ,会弹出一个创建活动的对话框,我们这次将活动命名为 SecondActivity ,并勾选 Generate Layout File ,给布局文件起名为 second_layout,但不要勾选 Launcher Activity 选项,如图 2.14 所示。

89144f780507e93cad6b125bd0977000.png

点击 Finish 完成创建,Android Studio 会为我们自动生成 SecondActivity.java 和 second_ layout.xml 这两个文件。不过自动生成的布局代码目前对你来说可能有些复杂,这里我们仍然还是使用最熟悉的 LinearLayout ,编辑 second_ layout.xml ,将里面的代码替换成如下内容:

<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>

我们还是定义了一个按钮,按钮上显示 Button 2。

然后 SecondActivity 中的代码已经自动生成了一部分,我们保持默认不变就好,如下所示:

public class SecondActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.second_layout);
    }

}

另外不要忘记,任何一个活动都是需要在 AndroidManifest.xml 中注册的,不过幸运的是,Android Studio 已经帮我们自动完成了,你可以打开 AndroidManifest.xml 瞧一瞧:

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <activity
        android:name=".FirstActivity"
        android:label="This is FirstActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name=".SecondActivity"></activity>
</application>

由于 SecondActivity 不是主活动,因此不需要配置 < intent-filter > 标签里的内容,注册活动的代码也简单了许多。现在第二个活动已经创建完成,剩下的问题就是如何去启动这第二个活动了,这里我们需要引入一个新的概念:【Intent】。

Intent 是 Android 程序中各组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,还可以在不同组件之间传递数据。Intent 一般可被用于启动活动、启动服务以及发送广播等场景,由于服务、广播等概念你暂时还未涉及,那么本章我们的目光无疑就锁定在了启动活动上面。

Intent 大致可以分为两种:显式 Intent 和 隐式 Intent ,我们先来看一下显式 Intent 如何使用。

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

修改 FirstActivity 中按钮的点击事件,代码如下所示:

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

我们首先构建出了一个 Intent,传入 FirstActivity.this 作为上下文,传入 SecondActivity.class 作为目标活动,这样我们的“意图”就非常明显了,即在 FirstActivity 这个活动的基础上打开 SecondActivity 这个活动。然后通过 startActivity() 方法来执行这个 Intent。

重新运行程序,在 FirstActivity 的界面点击一下按钮,结果如下图所示。

6f0c601b1ec46e4eb6ddcadce719270d.png

可以看到,我们已经成功启动 SecondActivity 这个活动了。如果你想要回到上一个活动怎么办呢?很简单,按下 Back 键就可以销毁当前活动,从而回到上一个活动了。

使用这种方式来启动活动,Intent 的“意图”非常明显,因此我们称之为 显式 Intent 。


使用隐式 Intent

相比于显式 Intent,隐式 Intent 则含蓄了许多,它并不明确指出我们想要启动哪一个活动,而是指定了一系列更为抽象的 action 和 category 等信息,然后交由系统去分析这个 Intent,并帮我们找出合适的活动去启动。

什么叫作合适的活动呢?简单来说就是可以响应我们这个隐式 Intent 的活动,那么目前 SecondActivity 可以响应什么样的隐式 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 中。

重新运行程序,在 FirstActivity 的界面点击一下按钮,你同样成功启动 SecondActivity 了。不同的是,这次你是使用了隐式 Intent 的方式来启动的,说明我们在 <activity> 标签下配置的 action 和 category 的内容已经生效了!

每个 Intent 中只能指定一个 action,但却能指定多个 category。目前我们的 Intent 中只有一个默认的 category,那么现在再来增加一个吧。

修改 FirstActivity 中按钮的点击事件,代码如下所示:

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

可以调用 Intent 中的 addCategory() 方法来添加一个 category,这里我们指定了一个自定义的 category,值为 com.example.activitytest.MY _CATEGORY。

现在重新运行程序,在 FirstActivity 的界面点击一下按钮,你会发现,程序崩溃了!这是你第一次遇到程序崩溃,可能会有些束手无策。别紧张,其实大多数的崩溃问题都是很好解决的,只要你善于分析。在 logcat 界面查看错误日志,你会看到如图 2.16 所示的错误信息。

8dd86b8d7bff6fd30d834867bdcc9385.png

错误信息中提醒我们,没有任何一个活动可以响应我们的 Intent,为什么呢?这是因为我们刚刚在 Intent 中新增了一个 category,而 SecondActivity 的 < intent-filter > 标签中并没有声明可以响应这个 category,所以就出现了没有任何活动可以响应该 Intent 的情况。现在我们在 < intent-filter > 中再添加一个 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>

再次重新运行程序,你就会发现一切都正常了。


更多隐式 Intent 的用法

上一节中,你掌握了通过隐式 Intent 来启动活动的方法,但实际上隐式 Intent 还有更多的内容需要你去了解,这一节我们就来展开介绍一下。

使用隐式 Intent,我们不仅可以启动自己程序内的活动,还可以启动其他程序的活动,这使得 Android 多个应用程序之间的功能共享成为了可能。比如说你的应用程序中需要展示一个网页,这时你没有必要自己去实现一个浏览器(事实上也不太可能),而是只需要调用系统的浏览器来打开这个网页就行了。

修改 FirstActivity 中按钮点击事件的代码,如下所示:

button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        Intent intent = new Intent(Intent.ACTION_VIEW);
        intent.setData(Uri.parse("百度一下,你就知道"));
        startActivity(intent);
    }
});

这里我们首先指定了 Intent 的 action 是 Intent.ACTION _VIEW,这是一个 Android 系统内置的动作,其常量值为 android.intent.action.VIEW 。然后通过 Uri.parse() 方法,将一个网址字符串解析成一个 Uri 对象,再调用 Intent 的 setData() 方法将这个 Uri 对象传递进去。

重新运行程序,在 FirstActivity 界面点击按钮就可以看到打开了系统浏览器,如下图所示。

75e4f71a02b8cda9207252530e077bf6.png

在上述代码中,可能你会对 setData() 部分感觉到陌生,这是我们前面没有讲到的。这个方法其实并不复杂,它接收一个 Uri 对象,主要用于指定当前 Intent 正在操作的数据,而这些数据通常都是以字符串的形式传入到 Uri.parse() 方法中解析产生的。

与此对应,我们还可以在 < intent-filter > 标签中再配置一个 <data> 标签,用于更精确地指定当前活动能够响应什么类型的数据。<data> 标签中主要可以配置以下内容。

android:scheme:用于指定数据的协议部分,如上例中的 http 部分。
android:host:用于指定数据的主机名部分,如上例中的 百度一下,你就知道 部分。
android:port:用于指定数据的端口部分,一般紧随在主机名之后。
android:path:用于指定主机名和端口之后的部分,如一段网址中跟在域名之后的内容。
android:mimeType:用于指定可以处理的数据类型,允许使用通配符的方式进行指定。

只有 <data> 标签中指定的内容和 Intent 中携带的 Data 完全一致时,当前活动才能够响应该 Intent。不过一般在 <data> 标签中都不会指定过多的内容,如上面浏览器示例中,其实只需要指定 android:scheme 为 http,就可以响应所有的 http 协议的 Intent 了。

为了让你能够更加直观地理解,我们来自己建立一个活动,让它也能响应打开网页的 Intent。

右击 com.example.activitytest 包 →New→Activity→ Empty Activity ,新建 ThirdActivity ,并勾选 Generate Layout File ,给布局文件起名为 third_layout,点击 Finish 完成创建。然后编辑 third_ layout.xml ,将里面的代码替换成如下内容:

<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 中的代码保持不变就可以了,最后在 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>

我们在 ThirdActivity 的 < intent-filter > 中配置了当前活动能够响应的 action 是 Intent.ACTION _VIEW 的常量值,而 category 则毫无疑问指定了默认的 category 值,另外在 <data> 标签中我们通过 android:scheme 指定了数据的协议必须是 http 协议,这样 ThirdActivity 应该就和浏览器一样,能够响应一个打开网页的 Intent 了。让我们运行一下程序试试吧,在 FirstActivity 的界面点击一下按钮,结果如下图所示。

4f852f8bdac2de6c58bea1c8607d209e.png

可以看到,系统自动弹出了一个列表,显示了目前能够响应这个 Intent 的所有程序。选择 Browser 还会像之前一样打开浏览器,并显示百度的主页,而如果选择了 ActivityTest ,则会启动 ThirdActivity 。JUST ONCE 表示只是这次使用选择的程序打开,ALWAYS 则表示以后一直都使用这次选择的程序打开。需要注意的是,虽然我们声明了 ThirdActivity 是可以响应打开网页的 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 的 action 是 Intent.ACTION _DIAL,这又是一个 Android 系统的内置动作。然后在 data 部分指定了协议是 tel,号码是 10086。重新运行一下程序,在 FirstActivity 的界面点击一下按钮,结果如下图所示。

f83a5105c089a4a9e595325c6c7215d8.png

向下一个活动传递数据

经过前面几节的学习,你已经对 Intent 有了一定的了解。不过到目前为止,我们都只是简单地使用 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 打印信息,如下图所示。

3ffb603a8136c78510c707a12fbe1d90.png

可以看到,我们在 SecondActivity 中成功得到了从 FirstActivity 传递过来的数据。


返回数据给上一个活动

既然可以传递数据给下一个活动,那么能不能够返回数据给上一个活动呢?答案是肯定的。不过不同的是,返回上一个活动只需要按一下 Back 键就可以了,并没有一个用于启动活动的 Intent 来传递数据。通过查阅文档你会发现,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 的打印信息,如下图所示。

6f7f274ef8a8a66c3cc84744a228b850.png

可以看到,SecondActivity 已经成功返回数据给 FirstActivity 了。

这时候你可能会问,如果用户在 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() 方法中的代码,我们在这里添加返回数据的逻辑就行了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值