Android - Intent

作者: AshengTan | 来源:发表于2016-05-08 19:25 被阅读894次

Intent

Intent 是 Android 中的各个组件之间交互的一种重要方式,它可以用来启动 Activity、Service 和 BroadcastReceiver,还可以在不同组件之间传递数据。

Intent Filter

  1. Intent Filter 就是用来注册 Activity 、 Service 和 Broadcast Receiver 具有能在某种数据上执行一个动作的能力。使用 Intent Filter,应用程序组件告诉 Android,它们能为其它程序的组件的动作请求提供服务,包括同一个程序的组件、本地的或第三方的应用程序。—— Android开发--Intent-filter属性详解
  2. Intent Filter 负责过滤掉组件本身无法响应和处理的 Intent,只将自己关心的 Intent 接收进来进行处理。—— IntentFilter

Intent 的七大属性

  1. ComponentName: 指定了 ComponentName 属性的 Intent已经明确了它将要启动哪个组件,这种 Intent 被称为显式 Intent;没有指定 ComponentName 属性的 Intent 被称为隐式 Intent。隐式Intent没有明确要启动哪个组件,应用会根据 Intent 指定的规则去启动符合条件的组件。
  2. Action: Action 属性用于指定要执行的动作,一个 Intent 只能设置一个 Action。
  3. Category: Category 属性为 Action 增加额外的附加类别信息。CATEGORY_LAUNCHER 意味着在加载程序的时候 Acticity 出现在最上面,而 CATEGORY_HOME 表示页面跳转到 HOME 界面。一个 Intent 可以添加多个 Category。
  4. Data: Data 属性通常用于向 Action 属性提供操作的数据。Data 属性的值是个 Uri 对象。
    Uri 的格式如下:scheme://host:port/path
    - scheme:用于指定数据的协议部分,如 http
    - host:用于指定数据的主机名部分,如 www.google.com
    - port:用于指定数据的端口部分,跟在主机后面,可以省略
    - path:用于指定主机名和端口之后的部分,通常是资源的路径,可以省略
  5. Type: Type 属性用于指定 Data 所指定的 Uri 对应的 MIME 类型。MIME 只要符合 “abc/xyz” 这样的字符串格式即可。
  6. Extras: Extras 属性用于保存需要传递的额外数据。
  7. Flag: Intent 可调用 addFlags() 方法来为 Intent 添加控制标记。

AndroidManifest.xml 中的 <intent-filter> 标签

<intent-filter> 标签可以包含以下三个元素:

  1. <action>: 一个 <intent-filter> 可以有一个或多个 <action> 用于过滤,到达的 Intent 只需要匹配其中一个 <action> 即可。
  2. <category>: 一个 <intent-filter> 中可以有多个 <category>,只有 Intent 中的所有 Category 都能匹配到 <intent-filter> 中的 <category> 时,Intent 才能通过检查。
  3. <data>: <data> 元素包含的内容为 Uri 和数据类型,<data> 元素中一般不会指定过多的内容。
    - Uri 格式见上面。
    - mimeType:用于指定可以处理的数据类型,可以省略。
  4. 更多内容,请看这篇文章 IntentFilter

Intent 的用法 - 启动组件


Intent 启动组件的方法可以分为两种,分别是显式 Intent 和 隐式 Intent。显式 Intent 必须明确指出要启动的是哪个组件;隐式 Intent 只须指出想要启动的组件的特征,系统就会自动启动符合该特征的组件。

启动组件 - 使用显式 Intent

这里我们以从一个 Activity 跳转到 另一个 Activity 为例。

1. XML 布局文件:

  1. activity_first.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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"
    android:gravity="center_horizontal"
    android:orientation="vertical"
    android:paddingBottom="16dp"
    android:paddingLeft="16dp"
    android:paddingRight="16dp"
    android:paddingTop="16dp"
    tools:context=".FirstActivity">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="这是第一个Activity" />

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="startSecondActivity"
        android:text="启动第二个Activity" />

</LinearLayout>
  1. activity_second.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center_horizontal"
    android:orientation="vertical"
    android:paddingBottom="16dp"
    android:paddingLeft="16dp"
    android:paddingRight="16dp"
    android:paddingTop="16dp">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="这是第二个Activity" />

</LinearLayout>

2. Java 代码:

  1. FirstActivity.java
public class FirstActivity extends AppCompatActivity {

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

    public void startSecondActivity(View view) {
        // 第一个参数为当前上下文,第二个参数为要启动的组件
        Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
        startActivity(intent);
    }

}
  1. SecondActivity.java
public class SecondActivity extends AppCompatActivity {

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

}
  1. 不要忘了在清单文件 AndroidManifest.xml 中注册第二个 Activity:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.monkeychan.intenttest">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme">
        <activity
            android:name=".FirstActivity"
            android:label="@string/app_name">
            <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>

</manifest>
  1. 效果演示:

点击按钮,启动第二个 Activity

启动组件 - 使用隐式 Intent

下面我们使用隐式 Intent 来实现上面的跳转效果。

1. 首先修改 AndroidManifest.xml 成如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.monkeychan.intenttest">

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

        <activity android:name=".SecondActivity">
            <intent-filter>
                <action android:name="net.monkeychan.intenttest.action.START" />
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>
    </application>

</manifest>

可以看到,我们给 SecondActivity 配置了 intent-filter,并在 intent-filter 下增加了 action 属性和 category 属性。

2. Java 代码,这里只须修改 FirstActivity.java:

public class FirstActivity extends AppCompatActivity {

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

    public void startSecondActivity(View view) {
        Intent intent = new Intent("net.monkeychan.intenttest.action.START");
        startActivity(intent);
    }

}
  1. 效果演示:

效果与上面使用显式 Intent 方式启动一样,只不过这里是使用隐式 Intent 方式实现罢了。

使用隐式 Intent 启动系统组件

使用隐式隐式 Intent 除了可以启动同一个应用程序的组件,还可以启动系统或第三方应用程序的组件。下面使用隐式 Intent 来启动系统的组件。

1. 启动拨号界面

  1. XML 布局文件,activity_main.xml:
<LinearLayout 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"
    android:orientation="vertical"
    android:paddingBottom="16dp"
    android:paddingLeft="16dp"
    android:paddingRight="16dp"
    android:paddingTop="16dp"
    tools:context=".MainActivity">

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="启动拨号界面"
        android:onClick="startDial"/>
</LinearLayout>
  1. Java 代码,MainActivity.java:
public class MainActivity extends AppCompatActivity {

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

   public void startDial(View view) {

       // 1. 创建一个 Intent 对象
       Intent intent = new Intent();

       // 2. 设置 Intent 对象的 Action
       // ACTION_DIAL 是 Intent 中定义的一个常量,查源码可知:
       // public static final String ACTION_DIAL = "android.intent.action.DIAL";
       // 这是 Android 中已经定义好的,事实上 Intent 中还有许多这样的常量
       intent.setAction(Intent.ACTION_DIAL);
       
       // 3. 跳转到对应的组件
       startActivity(intent);
   }

}
  1. AndroidManifest.xml 文件,记得声明权限,这里的权限是使用电话:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.monkeychan.intenttest">

    <!-- 声明该应用需要使用电话 -->
    <uses-permission android:name="android.permission.CALL_PHONE">

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

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

    </uses-permission>

</manifest>

上面的代码实现的功能是这样的:当我们点击按钮时,将会跳转到拨号界面。

  1. 效果演示:

点击按钮,将会跳转到拨号界面:

接下来我们对上面的程序进行修改,当点击按钮时,将会把我们要拨打的号码自动写好,我们只须按下拨号键就可以拨打了。

  1. XML 布局文件以及 AndroidManifest.xml 文件无须修改, Java 代码部分修改成如下:
public class MainActivity extends AppCompatActivity {

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

    public void startDial(View view) {

        // 1. 创建一个 Intent 对象
        Intent intent = new Intent();

        // 2. 设置 Intent 对象的 Action
        // ACTION_DIAL 是 Intent 中定义的一个常量,查源码可知:
        // public static final String ACTION_DIAL = "android.intent.action.DIAL";
        // 这是 Android 中已经定义好的,事实上 Intent 中还有许多这样的常量
        intent.setAction(Intent.ACTION_DIAL);

        // tel 是 Android 中已经定义好的,用于拨打电话,这里设置要拨打的号码为 10010
        Uri uri = Uri.parse("tel://10010");

        // 3. 调用 Intent 对象的 setData() 方法,该方法需要传入一个 Uri 类型的参数
        intent.setData(uri);

        // 4. 跳转到对应的组件
        startActivity(intent);
    }

}
  1. 效果演示:


点击按钮之后,号码将自动写好:

2. 发送短信

  1. XML 布局文件,activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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"
    android:orientation="vertical"
    android:paddingBottom="16dp"
    android:paddingLeft="16dp"
    android:paddingRight="16dp"
    android:paddingTop="16dp"
    tools:context=".MainActivity">

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="发送短信"
        android:onClick="sendSMS"/>
    
</LinearLayout>
  1. Java 代码,MainActivity.java:
public class MainActivity extends AppCompatActivity {

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

    public void sendSMS(View view) {

        // 1. 创建一个 Intent 对象
        Intent intent = new Intent();

        // 2. 设置 Intent 对象的 Action
        intent.setAction(Intent.ACTION_VIEW);

        // smsto 是 Android 中已经定义好的,用于发送,这里设置要发送的号码为 10010
        Uri uri = Uri.parse("smsto://10010");

        // 3. 设置 Intent 对象的 setData() 方法,该方法需要传入一个 Uri 类型的参数
        intent.setData(uri);

        // 4. 设置短信的内容,调用 Intent 对象的 putExtra() 方法,该方法需要传入一个键值对(key-value)
        // 其中 key 为 sms_body,是 Android 系统已经定义好的,系统的短信发送程序只能识别 sms_body
        intent.putExtra("sms_body", "CXYE");

        // 5. 跳转到对应的组件
        startActivity(intent);
    }

}
  1. 声明权限,这里的权限是发送短信,AndroidManifest.xml:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.monkeychan.intenttest">

    <uses-permission android:name="android.permission.SEND_SMS"/>

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <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>

</manifest>
  1. 效果演示:

点击按钮,启动发送短信程序,并将我们要发送的号码及内容传递给系统短信程序

点击发送按钮:

注意,这里使用的是虚拟机,实际上并没有给10010发送短信。

总结:使用 Intent 启动 Activity 的步骤

  1. 创建一个 Intent 对象;
    若是显式启动直接向构造方法里传入源 Activity 和 目的 Activity,然后进行第 3 步:
Intent intent = new Intent(OriActivity.this, DesActivity.class);

若是隐式启动直接调用无参的构造方法:

Intent intent = new Intent();
  1. 根据具体对 Intent 对象进行设置,如设置 Action、添加 Category 、设置 Data 等
    等;
intent.setAction("MyAction");
intent.addCategory("MyCategory");
intent.setData(Uri.parse("scheme://host"));
  1. 跳转到相应的 Activity。
startActivity(intent);

Intent 的用法 - 传递数据


Intent 除了可以用来启动各种组件之外,还可以用来在组件之间传递数据。

使用 Intent 在 Activity 之间传递数据

1. 传递简单数据

我们来做一个实现注册功能的程序:用户在注册界面填写用户名和密码,之后点击注册按钮,跳转到注册成功界面,该界面提示用户注册成功,并显示用户的注册信息。

1. XML 布局文件

  1. 注册界面,activity_signup.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   android:orientation="vertical"
   android:padding="16dp">

   <TableRow
       android:layout_width="match_parent"
       android:layout_height="wrap_content">

       <TextView
           android:id="@+id/tv_userName"
           android:layout_width="wrap_content"
           android:layout_height="wrap_content"
           android:text="用户名:" />

       <EditText
           android:id="@+id/et_userName"
           android:layout_width="match_parent"
           android:layout_height="wrap_content"
           android:hint="请输入用户名" />
   </TableRow>

   <TableRow
       android:layout_width="match_parent"
       android:layout_height="wrap_content">

       <TextView
           android:id="@+id/tv_userPassword"
           android:layout_width="wrap_content"
           android:layout_height="wrap_content"
           android:text="密码:" />

       <EditText
           android:id="@+id/et_userPassword"
           android:layout_width="match_parent"
           android:layout_height="wrap_content"
           android:hint="请输入密码" />
   </TableRow>

   <Button
       android:id="@+id/btn_signUp"
       android:layout_width="match_parent"
       android:layout_height="wrap_content"
       android:text="注册"
       android:onClick="signUp"/>

</LinearLayout>
  1. 注册成功界面,activity_signup_success.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="16dp">

    <TextView
        android:id="@+id/tv_userName"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/tv_userPassword"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</LinearLayout>

2. Java 代码

  1. 注册界面,SignUpActivity.java:
public class SignUpActivity extends AppCompatActivity {

    private EditText et_userName;
    private EditText et_userPassword;

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

        // 设置 title
        setTitle("注册界面");

        initView();

    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        et_userName = (EditText) findViewById(R.id.et_userName);
        et_userPassword = (EditText) findViewById(R.id.et_userPassword);
    }


    public void signUp(View view) {

        // 获取 EditText 里的数据
        String userName = et_userName.getText().toString();
        String userPassword = et_userPassword.getText().toString();

        // 1. 创建一个 Intent 对象,并传入将要启动的 Activity
        Intent intent = new Intent(SignUpActivity.this, SignUpSuccessActivity.class);

        // 2. 往 Intent 的对象里放入需要传递的数据
        intent.putExtra("userName", userName);
        intent.putExtra("userPassword", userPassword);

        // 3. 启动相应的组件,并将数据传递过去
        startActivity(intent);
    }

}
  1. 注册成功界面,SignUpSuccessActivity.java:
public class SignUpSuccessActivity extends AppCompatActivity {

    private TextView tv_userName;
    private TextView tv_userPassword;
    private Intent intent;

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

        // 设置 title
        setTitle("注册成功");

        initView();
        display();

    }

    // 该方法用来显示用户的注册信息
    private void display() {

        //1. 获取启动本 Activity 的 Intent
        intent = getIntent();

        // 2. 从 Intent 中取出数据,根据键取出相应的值
        // 注意,此键要与传递过来的键一致,否则会出现 NullPointerException 异常
        String userName = intent.getStringExtra("userName");
        String userPassword = intent.getStringExtra("userPassword");

        // 3. 显示用户的注册信息
        tv_userName.setText("您的用户名为:" + userName);
        tv_userPassword.setText("您的注册密码为:" + userPassword);

    }

    // 该方法用来实例化布局中的控件
    private void initView() {
        tv_userName = (TextView) findViewById(R.id.tv_userName);
        tv_userPassword = (TextView) findViewById(R.id.tv_userPassword);
    }

}

3. 最后,在 AndroidManifest.xml 中注册:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.monkeychan.intenttest">

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

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".SignUpSuccessActivity"></activity>
    </application>

</manifest>

4) 效果演示:

输入用户名和密码:

点击注册按钮:

返回数据给上一个 Activity

除了向下一个 Activity 传递数据,下一个 Activity 也可以向上一个 Activity 传递数据,具体步骤如下:

  1. 上一个 Activity 在跳转到相应的 Activity 时调用 startActivityForResult(Intent intent, int requestCode) 方法;
  2. 在下一个 Activity 调用 setResult(int resultCode, Intent data) 方法;
  3. 重写上一个 Activity 中的 onActivityResult(int requestCode, int resultCode, Intent data) 方法,在此方法中根据 resultCode 和 resultCode 对对应的返回数据进行处理。
    注意:requestCode 必须是唯一的,resultCode 也必须是唯一的,但两者可以相同。

修改上面的程序中 Java 部分的代码如下:

  1. SignUpActivity.java:
public class SignUpActivity extends AppCompatActivity {

    private EditText et_userName;
    private EditText et_userPassword;
    private Button btn_signUp;

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

        // 设置 title
        setTitle("注册界面");

        initView();

    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        et_userName = (EditText) findViewById(R.id.et_userName);
        et_userPassword = (EditText) findViewById(R.id.et_userPassword);
        btn_signUp = (Button) findViewById(R.id.btn_signUp);
    }


    public void signUp(View view) {

        // 获取 EditText 里的数据
        String userName = et_userName.getText().toString();
        String userPassword = et_userPassword.getText().toString();

        // 1. 创建一个 Intent 对象,并传入将要启动的目的 Activity
        Intent intent = new Intent(SignUpActivity.this, SignUpSuccessActivity.class);

        // 2. 往 Intent 的对象里放入需要传递的数据
        intent.putExtra("userName", userName);
        intent.putExtra("userPassword", userPassword);

        // 3. 启动相应的组件,将数据传递过去,并设置 requestCode
        startActivityForResult(intent, 0);
    }

    // 在此方法中根据 requestCode 和 resultCode 对对应的返回数据进行处理
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (requestCode == 0 && resultCode == 0) {
            // 将 Button 上的文字设置为返回的数据
            btn_signUp.setText(data.getStringExtra("signUpSuccess"));
            btn_signUp.setTextColor(Color.RED);
        }
    }

}
  1. SignUpSuccessActivity.java:
public class SignUpSuccessActivity extends AppCompatActivity {

    private TextView tv_userName;
    private TextView tv_userPassword;
    private Intent intent;

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

        // 设置 title
        setTitle("注册成功");

        initView();
        display();

    }

    // 该方法用来显示用户的注册信息
    private void display() {

        //1. 获取启动本 Activity 的 Intent
        intent = getIntent();

        // 2. 从 Intent 中取出数据,根据键取出相应的值
        // 注意,此键要与传递过来的键一致,否则会出现 NullPointerException 异常
        String userName = intent.getStringExtra("userName");
        String userPassword = intent.getStringExtra("userPassword");

        // 3. 显示用户的注册信息
        tv_userName.setText("您的用户名为:" + userName);
        tv_userPassword.setText("您的注册密码为:" + userPassword);

        // 设置返回的数据
        intent.putExtra("signUpSuccess", "注册成功");

        // 设置 resultCode 和返回的 Intent
        setResult(0, intent);

    }

    // 该方法用来实例化布局中的控件
    private void initView() {
        tv_userName = (TextView) findViewById(R.id.tv_userName);
        tv_userPassword = (TextView) findViewById(R.id.tv_userPassword);
    }

}
  1. 效果演示:

填写信息并点击提交注册按钮:

点击返回键:

可以看到,按钮上的文字即为返回的数据。

使用 Bundle 传递数据

除了 Intent,Bundle 也可以用来传递数据,其用法跟 Intent 差不多。将上面的程序修改成如下 (只须修改 SignUpActivity.java 和 SignUpSuccessActivity.java):

  1. SignUpActivity.java:
public class SignUpActivity extends AppCompatActivity {

    private EditText et_userName;
    private EditText et_userPassword;

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

        // 设置 title
        setTitle("注册界面");

        initView();

    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        et_userName = (EditText) findViewById(R.id.et_userName);
        et_userPassword = (EditText) findViewById(R.id.et_userPassword);
    }


    public void signUp(View view) {

        // 获取 EditText 里的数据
        String userName = et_userName.getText().toString();
        String userPassword = et_userPassword.getText().toString();

        // 1. 创建一个 Intent 对象,并传入将要启动的 Activity
        Intent intent = new Intent(SignUpActivity.this, SignUpSuccessActivity.class);

        // 2. 创建一个 Bundle 对象
        Bundle bundle = new Bundle();

        // 3. 往 Bundle 的对象里放入需要传递的数据
        bundle.putString("userName", userName);
        bundle.putString("userPassword", userPassword);

        // 4. 将 Bundle 对象放进 Intent 里
        intent.putExtras(bundle);

        // 5. 启动相应的组件,并将数据传递过去
        startActivity(intent);
    }

}
  1. SignUpSuccessActivity.java:
public class SignUpSuccessActivity extends AppCompatActivity {

    private TextView tv_userName;
    private TextView tv_userPassword;
    private Bundle bundle;

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

        // 设置 title
        setTitle("注册成功");

        initView();
        display();

    }

    // 该方法用来显示用户的注册信息
    private void display() {

        //1. 获取启动本 Activity 的 Intent 里的 Bundle
        bundle = getIntent().getExtras();

        // 2. 从 Bundle 中取出数据,根据键取出相应的值
        // 注意,此键要与传递过来的键一致,否则会出现 NullPointerException 异常
        String userName = bundle.getString("userName");
        String userPassword = bundle.getString("userPassword");

        // 3. 显示用户的注册信息
        tv_userName.setText("您的用户名为:" + userName);
        tv_userPassword.setText("您的注册密码为:" + userPassword);

    }

    // 该方法用来实例化布局中的控件
    private void initView() {
        tv_userName = (TextView) findViewById(R.id.tv_userName);
        tv_userPassword = (TextView) findViewById(R.id.tv_userPassword);
    }

}

从上面的代码可以看出,使用 Bundle 来传递数据跟使用 Intent 来传递数据差不多。实际上还是用的 Intent 来传递数据,只不过 Intent 并不直接将数据传递过去,而是传递一个 Bundle 对象,我们将要传递的数据封装在一个 Bundle 对象里,再使用 Intent 将 Bundle 对象传递过去。

  1. 效果演示:

效果跟上面的程序完全一样。

2. 传递对象

当要传递的数据较多时,我们可以把数据封装成一个对象,再把这个对象传递过去,条件是该对象必须实现 Serializable 接口或 Parcelable 接口。

  1. 对象实现 Serializable 接口

在使用内存方面,Serializable 效率比 Parcelable 低,但它能把数据存储在磁盘上。

a. XML 布局文件

信息提交页面,activity_info.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="16dp">

    <TableRow
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <TextView
            android:id="@+id/tv_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="姓名:" />

        <EditText
            android:id="@+id/et_name"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入姓名" />
    </TableRow>

    <TableRow
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <TextView
            android:id="@+id/tv_age"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="年龄:" />

        <EditText
            android:id="@+id/et_age"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入年龄" />
    </TableRow>

    <Button
        android:id="@+id/btn_summit"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="提交"
        android:onClick="summit"/>

</LinearLayout>

信息显示页面,activity_info_display.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="16dp">

    <TextView
        android:id="@+id/tv_name"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/tv_age"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</LinearLayout>

b. Java 代码

信息提交页面,InfoSummitActivity.java:

public class InfoSummitActivity extends AppCompatActivity {

    private EditText et_name;
    private EditText et_age;

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

        // 设置 title
        setTitle("提交信息");

        initView();
    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        et_name = (EditText) findViewById(R.id.et_name);
        et_age = (EditText) findViewById(R.id.et_age);
    }

    // 该方法用于提交用户信息,并跳转用户信息页面
    public void summit(View view) {

        // 获取 EditText 里的信息
        String name = et_name.getText().toString();
        int age = Integer.parseInt(et_age.getText().toString());

        // 1. 创建一个 Teacher 对象,并设置其属性
        Teacher teacher = new Teacher();
        teacher.setName(name);
        teacher.setAge(age);

        // 2. 创建一个 Intent 对象,指定要跳转的目的组件
        Intent intent = new Intent(InfoSummitActivity.this, InfoDisplayActivity.class);

        // 3. 将要传递的对象放入 Intent 中
        intent.putExtra("teacher", teacher);

        // 4. 跳转到对应的组件
        startActivity(intent);
    }
    
}

信息显示页面,InfoDisplayActivity.java:

public class InfoDisplayActivity extends AppCompatActivity {

    private TextView tv_name;
    private TextView tv_age;
    private Intent intent;

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

        // 设置 title
        setTitle("用户信息");

        initView();
        display();
    }

    // 该方法用于显示用户的信息
    private void display() {

        // 1. 获取启动本 Activity 的 Intent
        intent = getIntent();

        // 2. 创建一个 Teacher 对象,用来接收从 Intent 里取出的数据
        Teacher teacher = (Teacher) intent.getSerializableExtra("teacher");

        // 3. 设置数据到对应的控件上
        tv_name.setText("您的姓名是:" + teacher.getName());
        tv_age.setText("您的年龄为:" + teacher.getAge());
        // 注意,由于 setText 只能接受 CharSequence 类型的参数,
        // 而在这里我们的年龄是 int 类型的,如果写成:
        // tv_age.setText(teacher.getAge());
        // 点击按钮时会出现 Resources$NotFoundException,即资源未找到异常
        // 解决办法是加个空字符:tv_age.setText(teacher.getAge() + "");

    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        tv_name = (TextView) findViewById(R.id.tv_name);
        tv_age = (TextView) findViewById(R.id.tv_age);
    }

}

c. 效果演示:

输入信息,并点击提交按钮:

  1. 对象实现 Parcelable 接口
  1. 在使用内存方面,Parcelable 效率比 Serializable 高,但它不能将数据存储在磁盘上。
  2. 实现 Parcelable 接口必须重写 writeToParcel() 方法和 describeContents() 方法,并实例化静态内部对象 CREATOR 实现 Creator 接口。详情请看这篇文章 Android中Parcelable接口用法

a. XML 布局文件

信息提交页面,activity_info.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="16dp">

    <TableRow
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <TextView
            android:id="@+id/tv_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="姓名:" />

        <EditText
            android:id="@+id/et_name"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入姓名" />
    </TableRow>

    <TableRow
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <TextView
            android:id="@+id/tv_id"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="学号:" />

        <EditText
            android:id="@+id/et_id"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入学号" />
    </TableRow>

    <Button
        android:id="@+id/btn_summit"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="提交"
        android:onClick="summit"/>

</LinearLayout>

信息显示页面,activity_info_display.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="16dp">

    <TextView
        android:id="@+id/tv_name"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/tv_id"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</LinearLayout>

b. Java 代码

新建一个 Student 类,该类实现了 Parcelable 接口:

public class Student implements Parcelable {

    private String name;
    private int id;

    public Student() {

    }

    public Student(Parcel source) {
        name = source.readString();
        id = source.readInt();
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    // 此方法为内容描述,默认即可
    @Override
    public int describeContents() {
        return 0;
    }

    // 此方法用来写入数据
    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(this.name);
        dest.writeInt(this.id);
    }

    // 实例化 CREATOR 对象,实现 Creator 接口
    public static final Creator<Student> CREATOR = new Creator<Student>() {

        // 此方法用来读取传递过来的数据,注意读取的顺序要与写入的顺序一致
        @Override
        public Student createFromParcel(Parcel source) {
            return new Student(source);
        }

        // 此方法用来供外部类反序列化本类数组使用
        @Override
        public Student[] newArray(int size) {
            return new Student[size];
        }
    };

}

信息提交页面,InfoSummitActivity.java:

public class InfoSummitActivity extends AppCompatActivity {

    private EditText et_name;
    private EditText et_id;

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

        // 设置 title
        setTitle("提交信息");

        initView();
    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        et_name = (EditText) findViewById(R.id.et_name);
        et_id = (EditText) findViewById(R.id.et_id);
    }

    // 该方法用于提交用户信息,并跳转用户信息页面
    public void summit(View view) {

        // 获取 EditText 里的信息
        String name = et_name.getText().toString();
        int id = Integer.parseInt(et_id.getText().toString());

        // 1. 创建一个 Student 对象,并设置其属性
        Student student = new Student();
        student.setName(name);
        student.setId(id);

        // 2. 创建一个 Intent 对象,指定要跳转的目的组件
        Intent intent = new Intent(InfoSummitActivity.this, InfoDisplayActivity.class);

        // 3. 将要传递的对象放入 Intent 中
        intent.putExtra("student", student);

        // 4. 跳转到对应的组件
        startActivity(intent);
    }

}

信息显示页面,InfoDisplayActivity.java:

public class InfoDisplayActivity extends AppCompatActivity {

    private TextView tv_name;
    private TextView tv_id;
    private Intent intent;

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

        // 设置 title
        setTitle("用户信息");

        initView();
        display();
    }

    // 该方法用于显示用户的信息
    private void display() {

        // 1. 获取启动本 Activity 的 Intent
        intent = getIntent();

        // 2. 创建一个 Student 对象,用来接收从 Intent 里取出的数据
        Student student = intent.getParcelableExtra("student");

        // 3. 设置数据到对应的控件上
        tv_name.setText("您的姓名是:" + student.getName());
        tv_id.setText("您的学号是:" + student.getId());
        // 注意,由于 setText 只能接受 CharSequence 类型的参数,
        // 而在这里我们的年龄是 int 类型的,如果写成:
        // tv_id.setText(student.getId());
        // 点击按钮时会出现 Resources$NotFoundException,即资源未找到异常
        // 解决办法是加个空字符:tv_id.setText(student.getId() + "");

    }

    // 该方法用于实例化布局中的控件
    private void initView() {
        tv_name = (TextView) findViewById(R.id.tv_name);
        tv_id = (TextView) findViewById(R.id.tv_id);
    }

}

c. 效果演示:

填写信息,并点击提交按钮:

总结:使用 Intent 在 Activity 之间传递数据

  1. 在传递数据的地方创建一个 Intent 对象;
Intent intent = new Intent();
  1. 对 Intent 对象进行设置,可以传入简单的数据、对象 (包括 Bundle 对象),注意传入的参数键值对形式;
 - intent.putExtra("key", value);
 - Person person = new Person(); intent.putExtra("person", person);
 - Bundle bundle = new Bundle(); bundle.putXxx("key", value); intent.putExtra("bundle", bundle); // Xxx 为具体数据类型
  1. 在接收数据的地方创建一个 Intent 对象;
Intent intent = new Intent();
  1. 获取启动本 Activity 的 Intent;
intent = getIntent();
  1. 从 Intent 中取出数据,根据键取出相应的值,注意,此键要与传递过来的键一致,否则会出现 NullPointerException 异常;根据传递过来的值的类型,用对应的数据类型去接收。
 - String string = intent.getStringExtra("string");
 - Teacher teacher = (Teacher) intent.getSerializableExtra("teacher");
 - Student student = intent.getParcelableExtra("student");

参考资料:

相关文章

网友评论

    本文标题:Android - Intent

    本文链接:https://www.haomeiwen.com/subject/gawrrttx.html