Intent
Intent是Android程序中各组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,还可以在不同组件之间传递数据。Intent一般可用于启动Activity、启动Service以及发送广播等场景。Intent大致可以分为两种:显式Intent和隐式Intent。
新建一个活动
新建一个Activity活动,取名为SecondActivity,并且勾选Generate a Layout File,给布局文件取名为second_activity。
把自动生成的second_activity.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/button2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Button 2"
/>
</LinearLayout>
SecondActivity.java 需要修改(我的Android Studio新建的文件会加一些其他代码,应该是考拉版本的原因)
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.second_activity);
}
}
要记得检查AndroidManifest.xml文件中是否已经进行了注册,一般情况下Android Studio会自动帮我们注册。因为SecondActivity不是主活动所以不需要配置。
显示Intent
Intent有多个构造函数的重载,其中一个是Intent(Context packageContext, Class<?> cls)。这个构造函数接收两个参数:第一个参数Context要求提供一个启动Activity的上下文;第二个参数Class用于指定想要启动的目标Activity。
button1.setOnClickListener(new View.OnClickListener(){
@Override
public void onClick(View v) {
// 修改代码片段
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
startActivity(intent);
// ---------------
}
});
隐式Intent
通过在标签下配置的内容,可以指定当前Activity能够响应的action和category。
<activity
android:name=".SecondActivity"
android:exported="false" >
<intent-filter>
<action android:name="com.example.activitytest.ACTION_START" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
修改FirstActivity文件。
button1.setOnClickListener(new View.OnClickListener(){
@Override
public void onClick(View v) {
// 修改代码片段
Intent intent = new Intent("com.example.activitytest.ACTION_START");
// ---------------
startActivity(intent);
}
});
同多隐式Intent用法
使用隐式Intent,不仅可以启动自己程序内的Activity,还可以启动其他程序的Activity,这就使多个应用程序之间的功能共享成为了可能。
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对象传递进去。
还可以在标签中再配置一个标签,用于更精确地指定当前Activity能够响应的数据。
只有当标签中指定的内容和Intent中携带的Data完全一致时,当前Activity才能够响应该Intent。
先创建一个ThirdActivity,修改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/button3"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Button 3"
/>
</LinearLayout>
编辑注册文件AndroidManifest.xml。
<activity
android:name=".ThirdActivity"
android:exported="true" >
<intent-filter tools:ignore="AppLinkUrlError">
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<data android:scheme="http" />
</intent-filter>
</activity>
除了https协议外,我们还可以指定很多其他协议,比如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中提供了一系列putExtra()方法的重载,可以把我们想要传递的数据暂存在Intent中,在启动另一个Activity后,就可以把这些数据从Intent中取出来。putExtra()方法接收两个参数,第一个参数是键,用于后面从Intent中取值,第二个参数才是真正要传递的数据。
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);
}
});
在SecondActivity中将传递的数据取出。
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.second_activity);
Intent intent = getIntent();
String data = intent.getStringExtra("extra_data");
Log.d("SecondActivity", data);
}
}
向上一个活动返回数据
Activity中有一个startActivityForResult()方法是用来启动活动的,这个方法期望在活动销毁的时候能够返回一个结果给上一个活动。startActivityForResult()方法接收两个参数,第一个参数还是Intent,第二个参数是请求码,用于在之后的回调中判断数据的来源。startActivityForResult()方法在新版本中已经被官方弃用,但是版本都是向下兼容的,所以虽然被弃用了,但并不影响使用。
button1.setOnClickListener(new View.OnClickListener(){
@Override
public void onClick(View v) {
// 修改代码片段
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
startActivityForResult(intent, 1);
// ---------------
}
});
在SecondActivity中注册点击事件,并在点击事件中添加返回数据的逻辑。setResult()方法接收两个参数:第一个参数用于向上一个Activity返回处理结果,一般只使用RESULT_OK或RESULT_CANCELED这两个值;第二个参数则把带有数据的Intent传递回去。最后调用了finish()方法来销毁当前Activity。
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.second_activity);
/*Intent intent = getIntent();
String data = intent.getStringExtra("extra_data");
Log.d("SecondActivity", data);*/
Button button2 = (Button) findViewById(R.id.button2);
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();
}
});
}
}
SecondActivity被销毁之后会回调上一个Activity的onActivityResult()方法,因此我们需要在FirstActivity中重写这个方法来得到返回的数据,可以通过ctrl+O来创建。onActivityResult()方法带有3个参数:第一个参数requestCode,即我们在启动Activity时传入的请求码;第二个参数resultCode,即我们在返回数据时传入的处理结果;第三个参数data,即携带着返回数据的Intent。
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
switch(requestCode){
case 1:
if(resultCode == RESULT_OK){
String returnedData = data.getStringExtra("data_return");
Log.d("FirstActivity", returnedData);
}
break;
default:
}
}