使用Intent在活动之间穿梭(Intent初识)

来源:互联网 发布:网络摄像头直接接电脑 编辑:程序博客网 时间:2024/05/16 12:09

Intent英文意思:意图。

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

Intent的用法大致可以分为两种:显示Intent和隐式Intent

显示Intent:

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

代码实现:
activity_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent">    <Button        android:layout_width="match_parent"        android:layout_height="wrap_content"        android:text="点击,跳转到第二个活动"        android:id="@+id/btn_stump" /></RelativeLayout>
activity_second.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent">    <TextView        android:layout_width="match_parent"        android:layout_height="wrap_content"        android:text="我是第二个页面"        android:textColor="#f00"        android:textSize="16sp"        android:gravity="center" /></RelativeLayout>
SecondActivity.java
package com.example.intent;import android.app.Activity;import android.os.Bundle;public class SecondActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_second);}}
MainActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;public class MainActivity extends Activity {private Button btnStump;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);btnStump = (Button) findViewById(R.id.btn_stump);btnStump.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent(MainActivity.this,SecondActivity.class);startActivity(intent);}});}}
清单文件里的配置:
<?xml version="1.0" encoding="utf-8"?><manifest xmlns:android="http://schemas.android.com/apk/res/android"    package="com.example.intent"    android:versionCode="1"    android:versionName="1.0" >    <uses-sdk        android:minSdkVersion="8"        android:targetSdkVersion="16" />    <application        android:allowBackup="true"        android:icon="@drawable/ic_launcher"        android:label="@string/app_name"        android:theme="@style/AppTheme" >        <activity            android:name="com.example.intent.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>        <!-- 由于SecondActivity不是主活动,因此不需要配置<intent-filter>标签里的内容,注册活动的代码也是简单了许多。 -->        <activity android:name="com.example.intent.SecondActivity"/>    </application></manifest>
运行效果:
以上代码实现的思路:
我们首先构建出了一个Intent,传入MainActivity.this作为上下文,传入SecondActivity.class作为目标活动,这样我们的“意图”就非常明显了,即在MainActivity这个活动的基础上打开SecondActivity这个活动。然后通过startActivity()方法来执行这个Intent。
可以看到,我们已经成功启动SecondActivity这个活动了。如果你想要回到上一个活动怎么办呢?很简单,按下Back键就可以销毁当前活动,从而回到上一个活动了。
使用这种方式来启动活动,Intent的“意图”非常明显,因此我们称之为显示Intent。

隐式Intent

相比于显示Intent,隐式Intent则含蓄了许多,它并不明确指出我们想要启动哪一个活动,而是指定了一系列更为抽象的action和category等信息,然后交由系统去分析这个Intent,并帮助我们找出合适的活动去启动。
什么叫合适的活动呢?简单来说就是可以响应我们这个隐式Intent的活动,那么目前SecondAcivity可以响应什么样的隐式Intent呢?现在还没有。
通过在<activity>标签下配置<intent_filter>的内容,可以指定当前活动能够响应的action和category,打开AndroidMainFest.xml,添加如下代码:
<activity android:name="com.example.intent.SecondActivity" >    <intent-filter>        <action android:name="com.example.intent.ACTION_START"/>        <category android:name="android.intent.category.DEFAULT"/>    </intent-filter></activity>
在<action>标签中我们指明了当前活动可以响应com.example.intent.ACTION_START这个action,而<category>标签包含了一些附加信息,更精确地指明了当前的活动能够响应的Intent中还可能带有的category。只有<action>和<category>中的内容同时能偶匹配上Intent中指定的action和category时,这个活动才能响应该Intent。
修改MainActivity中按钮的点击事件,代码如下:
btnStump.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent("com.example.intent.ACTION_START");startActivity(intent);}});
可以看到,我们使用了Intent的另一个构造函数,直接将action的字符串传了进去,表明我们想要启动能够响应com.example.intent.ACTION_START这个action的活动。那前面不是说要<action>和<category>同时匹配上才能响应的吗?怎么没看到那里有指定category呢?这是因为android.intent.category.DEFAULT是一种默认的category,在调用startActivity()方法的时候会自动将这个category添加到Intent中。
重新运行程序,启动成功。不同的是,这次我们使用了隐式Intent的方式来启动的,说明我们在<activity>标签下配置的action和category的内容已经生效了!
每个Intent中只能指定一个action,但却能指定多个category。
目前我们的Intent中只有一个默认的category,那么现在再来增加一个吧。
btnStump.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent("com.example.intent.ACTION_START");intent.addCategory("com.example.intent.MY_CATEGORY");startActivity(intent);}});
可以调用Intent中的addCategory()方法来添加一个category,这里我们指定了一个自定义的category,值为com.example.intent.MY_CATEGORY。
现在重新运行程序,在MainActivity的界面点击一下按钮,你会发现,程序崩溃了!
android.content.ActivityNotFoundException: No Activity found to handle Intent { act=com.example.intent.ACTION_START cat=[com.example.intent.MY_CATEGORY] }
错误信息中提醒我们,没有任何一个活动可以响应我们的Intent,为什么呢?这是因为我们刚刚在Intent中新增了一个category,而SecondActivity的<intent-filter>标签中并没有声明可以响应这个category,所以就出现了没有任何活动可以响应该Intent的情况。现在我们在<intent-filter>中再添加一个category的声明。
<activity android:name="com.example.intent.SecondActivity" >    <intent-filter>        <action android:name="com.example.intent.ACTION_START"/>            <category android:name="android.intent.category.DEFAULT"/>            <category android:name="com.example.intent.MY_CATEGORY"/>    </intent-filter></activity>
重新运行,一切正常。

更多隐式Intent的用法:

使用隐式Intent,我们不仅可以启动自己程序内的活动,还可以启动其他程序的活动,这使得Android多个应用程序之间的功能共享成为了可能。比如说你的应用程序中需要展示一个网页,这时没有必要自己去实现一个浏览器,而是只需要调用系统的浏览器来打开这个网页就行了。
MainActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.net.Uri;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;public class MainActivity extends Activity {private Button btnStump;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);btnStump = (Button) findViewById(R.id.btn_stump);btnStump.setOnClickListener(new OnClickListener() {@Overridepublic 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 data)方法,它接收一个Uri对象,主要用于指定当前Intent正在操作的数据,而这些数据通常都是以字符串的形式传入到Uri.parse()方法中解析产生的。
与此对应,我们还可以在<intent-filter>标签中再配置一个<data>标签,用于更精确地指定当前活动能够响应什么类型的数据。
<data>标签中主要可以配置以下内容。
  1. android:scheme
    • 用于指定数据的协议部分。如上例中的http部分
  2. android:host
    • 用于指定数据的主机名部分。如上列中的www.baidu,com部分
  3. android:port
    • 用于指定数据的端口部分,一般紧随在主机名之后。
  4. android:path
    • 用于指定主机名和端口之后的部分。如一段网址中跟在域名之后的内容。
  5. android:mimeType
    • 用于指定可以处理的数据类型,允许使用通配符的方式进行指定。
只有<data>标签中指定的内容和Intent中携带的Data完全一致时,当前活动才能够响应该Intent。不过一般在<data>标签中都不会指定过多的内容。如上面浏览器示例中,其实只需要指定android:scheme为http,就可以响应所有的http协议的Intent了。
除了http协议外,我们还可以指定很多其他协议,比如geo表示显示地理位置、tel表示拨打电话。下面的代码展示了如何在我们的程序中调用系统拨号界面。
MainActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.net.Uri;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;public class MainActivity extends Activity {private Button btnStump;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);btnStump = (Button) findViewById(R.id.btn_stump);btnStump.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {//首先指定了Intent的action是Intent.ACTION_DIAL,这又是一个Adnroid系统的内置动作。Intent intent = new Intent(Intent.ACTION_DIAL);//然后在data部分指定了协议是telintent.setData(Uri.parse("tel:10086"));startActivity(intent);}});}}
运行效果:

向下一个活动传递数据:

其实Intent还可以在启动活动的时候传递数据,Intent中提供了一系列putExtra()方法的重载,可以把我们想要传递的数据暂存在Intent中,启动了另一个活动后,只需要把这些数据再从Intent中取出就可以了。
比如我说MainActivity中有一个字符串,现在想要把这个字符串传递到SecondActivity中。
MainActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;public class MainActivity extends Activity {private Button btnStump;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);btnStump = (Button) findViewById(R.id.btn_stump);btnStump.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {String data = "Hello SecondActivity";//使用显式Intent的方式来启动SecondActivityIntent intent = new Intent(MainActivity.this, SecondActivity.class);//通过putExtra()方法传递了一个字符串/** * 注意这里putExtra()方法接收两个参数,第一个参数是键,用于后面从Intent中取值, * 第二个参数才是真正要传递的数据 */intent.putExtra("extra_data", data);startActivity(intent);}});}}
SecondActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.widget.TextView;public class SecondActivity extends Activity {private TextView tvShowText;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_second);tvShowText = (TextView) findViewById(R.id.tv_showtext);//通过getIntent()方法获取到用于启动SecondActivity的IntentIntent intent = getIntent();//调用getStringExtra()方法,传入相应的键值,就可以得到传递的数据了。/** * 这里由于我们传递的是字符串,所以使用getStringExtra()方法来获取传递的数据, * 如果传递的是整型数据,则使用getIntExtra()方法,传递的是布尔型数据,则使用getBooleanExtra()方法,以此类推。 */String data = intent.getStringExtra("extra_data");tvShowText.setText(data);}}
运行结果:

返回数据给上一个活动

既然可以传递数据给下一个活动,那么能不能够返回数据给上一个活动呢?答案是肯定的。不过不同的是,返回上一个活动祝需要按一下back键就可以了,并没有一个用于启动活动Intent来传递数据。通过查看官方文档你会发现,Activity中还有一个startActivityForResult()方法也是用于启动活动的,但这个方法期望在活动销毁的时候能够返回一个结果给上一个活动。
startActivityForResult()方法接收两个参数,第一个参数还是Intent,第二个参数是请求码,用于在之后的回调中判断数据的来源。
MainActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;import android.widget.TextView;public class MainActivity extends Activity {private Button btnStump;private TextView tv1;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);tv1 = (TextView) findViewById(R.id.tv1);btnStump = (Button) findViewById(R.id.btn_stump);btnStump.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent(MainActivity.this,SecondActivity.class);/** * Intent intent:意图对象 int * requestCode:请求码。请求码只要是一个唯一值就可以了,这里传入了1。 */// 使用startActivityForResult(Intent intent, int requestCode)方法启动SecondActivitystartActivityForResult(intent, 1);}});}/** * 由于我们使用startActivityForResult()方法来启动SecondActivity的, * 在SecondActivity被销毁之后会回调上一个活动的onActivityResult()方法, 也就是回调该方法。 *//** * int requestCode:我们在启动活动时传入的请求码  * int resultCode:我们在返回数据时传入的处理结果码 Intent * data:携带这返回数据的Intent */@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);/** * 思路分析: 由于在一个活动中有可能调用startActivityForResult()方法去启动很多不同的活动, * 每一个活动返回的数据都会回调到onActivityResult()这个方法中, * 因此我首先要做的是通过检查requestCode的值来判断数据来源 * 。确定数据是从SecondActivity返回的之后,我们再通过resultCode的值来潘大U呢处理结果 * 是否成功。最后从data中取值显示。这样就完成了向上一个活动返回数据的工作。 */switch (requestCode) {case 1:if (resultCode == RESULT_OK) {String returnedData = data.getStringExtra("data_return");tv1.setText("返回的数据为:" + returnedData);}break;default:break;}}}
SecondActivity.java
package com.example.intent;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;public class SecondActivity extends Activity {private Button button;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_second);button = (Button) findViewById(R.id.button1);button.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {//1.构建了一个Intent,只不过这个Intent仅仅是用于传递数据而已,它没有指定任何的"意图"。Intent intent = new Intent();//2.把要传递的数据存放在Intent中intent.putExtra("data_return", "Hello MainActivity");//3.调用setResult(int resultCode, Intent data)方法/**这个方法非常重要,是专门用于向上一个活动返回数据的。 * int resultCode:结果码。用于向上一个活动返回处理结果,一般只是用RESULT_OK或RESULT_CANCELED这两个值。 * Intent data:把带有数据的Intent传递回去。 */setResult(RESULT_OK, intent);//4.销毁当前活动finish();}});}}
运行结果
可以看到,SecondActivity已经成功返回数据给MainActvity了。
但是,存在一个小小的bug.
如果用户在SecondActivity中并不是通过点击按钮,而是通过按下Back键回到MainActivity,这样数据不就没法返回了码?没错,不过这种情况也很好处理,我们可以通过重写
onBackPressed()方法来解决。
在SecondActivity.java中增加这个方法
@Overridepublic void onBackPressed() {Intent intent = new Intent();intent.putExtra("data_return", "Hello MainActivity");setResult(RESULT_OK,intent);finish();}
这样的话,当用户按下Back键,就会去执行onBackPressed方法中的代码。


0 0
原创粉丝点击