4、向下一个活动传递数据
之前我们只是简单地使用Intent来启动一个活动,其实Intent还可以在启动活动的时候传递数据的。
在启动活动时传递数据的思路很简单,Intent中提供了一系列putExtra()方法的重载,可以把我们想要传递的数据暂存在Intent中,启动了另一个活动后,只需要把这些数据再从Intent中取出就可以了。
比如说FirstActivity中有一个字符串,现在想把这个字符串传递到SecondActivity中,你就可以这样编写:
btn1.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.activity_second); Intent intent = getIntent(); String data = intent.getStringExtra("extra_data"); Log.d("SecondActivity", data); } }
首先可以通过getIntent()方法获取到用于启动SecondActivity的Intent,然后调用getStringExtra()方法,传入相应的键值,就可以得到传递的数据了。这里由于我们传递的是字符串,所以使用getStringExtra()方法来获取传递的数据,如果传递的是整型数据,则使用getIntExtra()方法,传递的是布尔型数据,则使用getBooleanExtra()方法,以此类推。
重新运行程序,在FirstActivity的界面点击一下按钮会跳转到SecondActivity,查看LogCat打印信息,如图所示。
可以看到,我们在SecondActivity中成功得到了从FirstActivity传递过来的数据。
另外,如果我们传递的数据比较多,我们还可以使用Bundle来将数据打包一下。
Bundle是一个最终类,它的说明:A mapping from String keys to various values.(一个字符串类型的键到各种类型的值的映射),通俗说,就是一个键值对的映射。它提供了一系列的putXXX()方法用于存放数据,提供了一系列的getXXX()方法用于获取数据,另外,还有一个get()方法,可以根据键值获取Object类型的对象,使用这个方法就可以获取任意类型的数据了。
我们修改FirstActivity.java中的代码:
btn1.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(FirstActivity.this, SecondActivity.class); Bundle bundle = new Bundle(); bundle.putString("name", "张三"); bundle.putInt("age", 32); bundle.putBoolean("male", true); intent.putExtras(bundle); startActivity(intent); } });
修改SecondActivity.java中的代码:
public class SecondActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_second); Intent intent = getIntent(); Bundle bundle = intent.getExtras(); Log.d("SecondActivity", "name = " + bundle.getString("name")); Log.d("SecondActivity", "age = " + bundle.getInt("age")); Log.d("SecondActivity", "male = " + bundle.getBoolean("male")); } }
我们如果想在第一个活动中自定义信息,然后点击按钮,将信息传递到第二个活动的界面上呢?
自己完成这个练习试试。
5、返回数据给上一个活动
我们可以传递数据给下一个活动,也肯定可以返回数据给上一个活动。
我们先来通过SecondActivity中的按钮2来返回数据给FirstActivity。
那么FirstActivity如何知道我们会返回数据给它呢?这时我们就需要用到Activity中的另一个能够启动活动的方法----startActivityForResult()。
startActivityForResult()方法接收两个参数,第一个参数还是Intent,第二个参数是请求码(int型),用于在之后的回调中判断数据的来源。
修改FirstActivity中按钮的点击事件,代码如下所示:
btn1.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); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_second); Button btn2 = (Button) findViewById(R.id.btn2); btn2.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; } }
onActivityResult()方法带有三个参数,第一个参数requestCode,即我们在启动活动时传入的请求码。
第二个参数resultCode,即我们在返回数据时传入的处理结果。
第三个参数data,即携带着返回数据的Intent。
由于在一个活动中有可能调用startActivityForResult()方法去启动很多不同的活动,每一个活动返回的数据都会回调到onActivityResult()这个方法中,因此我们首先要做的就是通过检查requestCode的值来判断数据来源。
确定数据是从SecondActivity返回的之后,我们再通过resultCode的值来判断处理结果是否成功。
最后从data中取值并打印出来,这样就完成了向上一个活动返回数据的工作。
重新运行程序,在FirstActivity的界面点击按钮会打开SecondActivity,然后在SecondActivity界面点击按钮2会回到FirstActivity,这时查看LogCat的打印信息,如图所示。
可以看到,SecondActivity已经成功返回数据给FirstActivity了。
问题来了:如果用户在SecondActivity中并不是通过点击按钮,而是通过按下Back键回到FirstActivity,这样数据不就没法返回了吗?没错,不过这种情况还是很好处理的,我们可以通过重写onBackPressed()方法来解决这个问题,代码如下所示:
@Override public void onBackPressed() { Intent intent = new Intent(); intent.putExtra("data_return", "Back Return"); setResult(RESULT_OK, intent); finish(); }
这样的话,当用户按下Back键,就会去执行onBackPressed()方法中的代码,我们在这里添加返回数据的逻辑就行了。
【注意】
在AndroidX的Activity1.2.0-alpha02和Fragment1.3.0-alpha02开始之后,startActivityForResult()被标注弃用,推荐使用registerForActivityResult()方法。