引言
我们知道,如果想打开一个新的Activity我们可以使用startActivity方法。今天我们介绍的startActivityForResult不仅可以打开全新的Activity,而且当新的Activity关闭后,父Activity可以接收到新窗口设置的值。这篇文章我们就来介绍下startActivityForResult和setResult这两个方法。下面来看例子吧。
实例
startActivityForResult方法
我们来看一个简单的例子。这个例子的MainActivity上有两个按钮,点击这两个按钮都会打开一个全新的界面SecodeActivity,SecodeActivity在退出时会向MainActivity传递数据。我们首先来看下MainActivity的代码:
package com.example.dreamgong.apprunresearch; import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast; public class MainActivity extends Activity { private Button mButton1;
private Button mButton2; private static final int REQUESTCODE1=0x0001;
private static final int REQUESTCODE2=0x0002; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); initControl();
} private void initControl() {
mButton1=(Button)findViewById(R.id.button);
mButton1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(getApplicationContext(),SecondActivity.class);
intent.putExtra("KEY1", "BTN1MSG1");
//打开新的Activity并且接受Activity的返回值
startActivityForResult(intent, REQUESTCODE1);
}
}); mButton2=(Button)findViewById(R.id.button2);
mButton2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(getApplicationContext(),SecondActivity.class);
intent.putExtra("KEY1", "BTN2MSG2");
//打开新的Activity并且接受Activity的返回值
startActivityForResult(intent, REQUESTCODE2);
}
});
} /**
* 当新打开的界面退出,处理返回的数据
* @param requestCode
* @param resultCode
* @param data
*/ @Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if(resultCode==Activity.RESULT_OK) {
if(requestCode==REQUESTCODE1){
String msg=data.getStringExtra("Second");
msg+="这是第一个按钮点击跳转的";
Toast.makeText(getApplicationContext(), msg, Toast.LENGTH_LONG).show();
}
if(requestCode==REQUESTCODE2){
String msg=data.getStringExtra("Second");
msg+="这是第二个按钮点击跳转的";
Toast.makeText(getApplicationContext(), msg, Toast.LENGTH_LONG).show();
}
}
} }
在代码中我们看到initControl方法主要负责初始化控件以及设置点击事件。我们看到两个按钮都设置了事件处理函数,都是跳转到SecondActivity。只是传递给SecondActivity界面的值不一致。我们也看到startActivityForResult方法的第二个参数是int类型的,需要传递RequestCode。
RequestCode的作用:
RequestCode的值是根据业务需要由自已设定,用于标识请求来源。例如:一个Activity有两个按钮,点击这两个按钮都会打开同一个Activity,不管是那个按钮打开新Activity,当这个新Activity关闭后,系统都会调用前面Activity的onActivityResult(int requestCode, int resultCode, Intent data)方法。在onActivityResult()方法如果需要知道新Activity是由那个按钮打开的。我们看到在onActivityResult方法中我们就是通过RequestCode来区分是哪一个按钮打开了新的界面,我们可以编写相应的业务代码。
setResult方法
下面我们来看SecondActivity中的逻辑代码,我们知道SecondActivity方法在关闭时需要向MainActivity方法中传递数据。怎么传递呢?我们结合Activity的生命周期来探究。
我们先来看SecondActivity退出,MainActivity重新呈现时,这一流程方法的调用过程。B代表新打开的SecondActivity,A表示MainActivity。整个过程如下:
B---onPause
A---onActivityResult
A---onRestart
A---onStart
A---onResume
B---onStop
B---onDestroy
从上面过程可以看出,首先是B处于Pause 状态,然后等待A执行 onRestart——> onStart ——〉onResume,然后才是B 的onStop——>onDestroy,而A的 onActivityResult() 需要在B的onPause之后,A的onRestart之前这中间调用,所以B中的setResult()函数应该放在B的onPause之前调用。
另外我试验了一下,如果把setResult()放在 B 的 onPause() 里面调用,结果仍然是无效的。
那么setResult()应该在什么时候调用呢?从源码可以看出,Activity返回result是在被finish的时候,也就是说调用setResult()方法必须在finish()之前。所以在onPause、onStop、onDestroy方法中调用setResult()也有可能不会返回成功,因为这些方法调用不一定是在finish之前的。我们来详细看下源码:
setResult()方法的源码:
public final void setResult(int resultCode, Intent data) {
synchronized (this) {
mResultCode = resultCode;
mResultData = data;
}
}
finish()方法的源码:
private void finish(boolean finishTask) {
if (mParent == null) {
int resultCode;
Intent resultData;
synchronized (this) {
resultCode = mResultCode;
resultData = mResultData;
}
if (false) Log.v(TAG, "Finishing self: token=" + mToken);
try {
if (resultData != null) {
resultData.prepareToLeaveProcess();
}
if (ActivityManagerNative.getDefault()
.finishActivity(mToken, resultCode, resultData, finishTask)) {
mFinished = true;
}
} catch (RemoteException e) {
// Empty
}
} else {
mParent.finishFromChild(this);
}
}
看代码的6,7行mResultCode是必须先经过setResult方法进行赋值的。所以setResult方法必须在finish方法之前调用!!!
实际应用场景
在实际使用中我们使用setResult方法一般都借助onBackPressed()方法或者点击事件。看下面的例子:
1、按Back键(回退键):
按BACK键从一个Activity退出来的,一按BACK,android就会自动调用Activity的finish()方法,方法:重写onBackPressed()方法,捕获BACK事件,捕获到之后先setResult。代码如下:
@Override
public void onBackPressed() { Intent intent=new Intent();
intent.putExtra("Second","HELLO WORLD!!!");
setResult(Activity.RESULT_OK,intent);
finish();
super.onBackPressed(); }
2、按点击事件中显式的调用finish()
这时候整个执行流程如下:
B---onBackPressed
B---finish
B---onPause
A---onActivityResult
A---onRestart
A---onStart
A---onResume
B---onStop
B---onDestroy