不管创建多少活动,方法都是一样的,但是点击应用图标以后,进入的是应用的主活动,怎么才能由主活动跳转到其他活动呢?这就要用到intent啦!
intent是Android程序中各组件进行交互的一种重要方式,一般可用于启动活动、启动服务、发送广播等场景。大致可分为两种:显式intent和隐式intent。
显式intent
我们再创建一个活动SecondActivity,AS将自动生成SecondActivity.java和second_layout.xml这两个文件,我们依旧定义一个按钮Button2
<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/second_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Button 2"
/>
</LinearLayout>
一定记得任何一个活动都是需要在AndroidManifest.xml中注册的,此时AS已经帮我们自动完成了。
此时第二个活动已经创建,那么要怎么启动呢?intent有多个构造函数的重载,其中一个是intent(Context packageContext,Class<?>cls),第一个参数context要求提供一个启动活动的上下文,第二个参数class指定想要启动的目标活动。activity类提供了一个startActivity()方法,此方法专门用于启动活动。修改FirstActivity中的按钮点击事件:
button1.setOnClickListener(new View.OnClickListener(){
@Override
public void onClick(View v){
Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
startActivity(intent);
}
});
重新运行程序,点击button1,结果如下:
使用这种方式启动活动,intent的意图非常明显,我们称之为显式intent。
隐式intent
相比于显式,隐式则含蓄了许多,它指定了一系列更为抽象的action和category等信息,让系统去分析intent,帮我们找出合适的活动去启动。
配置以下内容,可以指定当前活动能够响应的action和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>
只有< action>和< category>中的内容同时能够匹配上intent中指定的action和category时,这个活动才能响应该intent。
此时我们使用intent的另一个构造函数,将action的字符串传进去,而指定的category在哪里呢?事实上android.intent.DEFAULT是一种默认的category,在调用startActivity()方法时会自动将这个category添加到intent中。
每个intent只能指定一个action,却可以指定多个category,可以调用intent中的addCategory()方法来添加一个category,代码如下:
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);
}
});
重新运行程序,你同样成功的启动了SecondActivity,只不过这次是使用隐式的!
更多隐式intent的用法
使用隐式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);
}
});
通过uri.parse()方法,将一个网址字符解析成一个uri对象,在调用intent的setData()方法将这个uri对象传递进去。
重新运行程序,点击button1跳转到该网址
向下一个活动传递数据
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);
}
});
注意这里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()方法,传入相应的键值,就可以得到传递的数据了。
运行程序,在FirstActivity界面点击按钮跳到SecondActivity,查看log cat打印信息:
log cat中显示如下
返回数据给上一个活动
返回上一个活动只需要按一下back键,Activity中有一个方法期望在活动销毁的时候能够返回一个结果给上一个活动,这个方法就是startActivityForResult(),该方法接收两个参数,第一个参数是intent,第二个参数是请求码,用于在之后的回调中判断数据的来源
button1.setOnClickListener(new View.OnClickListener(){
@Override
public void onClick(View v){
Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
startActivityForResult(intent,1);
}
});
请求码只要是一个唯一的值就可以,这里传入了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();
}
});
}
}
setResult()方法接收两个参数,第一个参数用于向上一个活动给返回处理结果,一般只使用RESULT_OK或者RESULT_CANCELED,第二个参数则把带有数据的intent传回去,然后调用了finish()方法销毁活动。
由于我们是使用startActivityForResult方法启动SecongActivity的,在SecongActivity被销毁之后会回调上一个活动的onActivityResult()方法,所以我们在主活动中要重写这个方法:
@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。
如果不是按按钮返回上一个活动,而是按back键,这样数据还会返回吗?我们在SeconfActivity中重写onBackPressed()方法就可以了
@Override
public void onBackPressed() {
Intent intent = new Intent();
intent.putExtra("data_return","Hello FirstActivity");
setResult(RESULT_OK);
finish();
}
这样按下back键,就会去执行此方法中的代码了
运行程序
出错了,什么原因呢?我查了一下午都没找到。。。。报错是这样,什么空指针,网上找了许多都不对。。。
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.activitytest/com.example.activitytest.SecondActivity}:
java.lang.NullPointerException: Attempt to invoke virtual method ‘void android.widget.Button.setOnClickListener(android.view.View$OnClickListener)’ on a null object reference