我有两个活动:主要活动和子活动。
当我在主活动中按下按钮时,将启动子活动。
现在我想将一些数据发送回主屏幕。我使用了 Bundle 类,但它不起作用。它会引发一些运行时异常。
有什么解决办法吗?
我有两个活动:主要活动和子活动。
当我在主活动中按下按钮时,将启动子活动。
现在我想将一些数据发送回主屏幕。我使用了 Bundle 类,但它不起作用。它会引发一些运行时异常。
有什么解决办法吗?
根据具体情况,有几种方法可以实现您想要的。
最常见的场景(听起来就是这样)是子 Activity 用于获取用户输入 - 例如从列表中选择联系人或在对话框中输入数据。在这种情况下,您应该使用startActivityForResult
来启动您的子活动。
这提供了一个管道,用于使用setResult
. setResult 方法接受一个 int 结果值和一个 Intent,该 Intent 被传回调用的 Activity。
Intent resultIntent = new Intent();
// TODO Add extras or a data URI to this intent as appropriate.
resultIntent.putExtra("some_key", "String data");
setResult(Activity.RESULT_OK, resultIntent);
finish();
在调用 Activity override 中访问返回的数据onActivityResult
。requestCode对应startActivityForResult
调用中传入的整数,而resultCode和data Intent是从子Activity返回的。
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch(requestCode) {
case (MY_CHILD_ACTIVITY) : {
if (resultCode == Activity.RESULT_OK) {
// TODO Extract the data returned from the child Activity.
String returnValue = data.getStringExtra("some_key");
}
break;
}
}
}
活动 1 使用startActivityForResult:
startActivityForResult(ActivityTwo, ActivityTwoRequestCode);
Activity 2 已启动,您可以执行操作,要关闭 Activity,请执行以下操作:
Intent output = new Intent();
output.putExtra(ActivityOne.Number1Code, num1);
output.putExtra(ActivityOne.Number2Code, num2);
setResult(RESULT_OK, output);
finish();
活动 1 - 从上一个活动返回将调用onActivityResult:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == ActivityTwoRequestCode && resultCode == RESULT_OK && data != null) {
num1 = data.getIntExtra(Number1Code);
num2 = data.getIntExtra(Number2Code);
}
}
更新:回答 Seenu69 的评论,在活动二中,
int result = Integer.parse(EditText1.getText().toString())
+ Integer.parse(EditText2.getText().toString());
output.putExtra(ActivityOne.KEY_RESULT, result);
然后在活动一中,
int result = data.getExtra(KEY_RESULT);
它帮助我在上下文中看待事物。这是一个用于发回数据的完整简单项目。这里没有提供 xml 布局文件,而是提供图像。
startActivityForResult
,为其提供任意结果代码。onActivityResult
。当第二个活动完成时调用它。您可以通过检查请求代码确保它实际上是第二个活动。(当您从同一个主要活动开始多个不同活动时,这很有用。)Intent
。使用键值对提取数据。MainActivity.java
public class MainActivity extends AppCompatActivity {
private static final int SECOND_ACTIVITY_REQUEST_CODE = 0;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
// "Go to Second Activity" button click
public void onButtonClick(View view) {
// Start the SecondActivity
Intent intent = new Intent(this, SecondActivity.class);
startActivityForResult(intent, SECOND_ACTIVITY_REQUEST_CODE);
}
// This method is called when the second activity finishes
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
// Check that it is the SecondActivity with an OK result
if (requestCode == SECOND_ACTIVITY_REQUEST_CODE) {
if (resultCode == RESULT_OK) {
// Get String data from Intent
String returnString = data.getStringExtra("keyName");
// Set text view with string
TextView textView = (TextView) findViewById(R.id.textView);
textView.setText(returnString);
}
}
}
}
Intent
. 数据存储在Intent
使用键值对中。RESULT_OK
并添加保存数据的意图。finish()
以关闭第二个活动。SecondActivity.java
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
}
// "Send text back" button click
public void onButtonClick(View view) {
// Get the text from the EditText
EditText editText = (EditText) findViewById(R.id.editText);
String stringToPassBack = editText.getText().toString();
// Put the String to pass back into an Intent and close this activity
Intent intent = new Intent();
intent.putExtra("keyName", stringToPassBack);
setResult(RESULT_OK, intent);
finish();
}
}
RESULT_OK
. 只需使用全名:Activity.RESULT_OK
.FirstActivity 使用 startActivityForResult:
Intent intent = new Intent(MainActivity.this,SecondActivity.class);
startActivityForResult(intent, int requestCode); // suppose requestCode == 2
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == 2)
{
String message=data.getStringExtra("MESSAGE");
}
}
在 SecondActivity 调用 setResult() onClick 事件或 onBackPressed()
Intent intent=new Intent();
intent.putExtra("MESSAGE",message);
setResult(Activity.RESULT_OK, intent);
使用 startActivityForResult() 方法调用来调用子活动 Intent
这里有一个例子:http: //developer.android.com/training/notepad/notepad-ex2.html
并在“从屏幕返回结果”中:http: //developer.android.com/guide/faq/commontasks.html#opennewscreen
2021 年 3 月更新
与Activity v1.2.0和Fragment v1.3.0Activity Result APIs
一样,已经引入了新功能。
活动结果 API 提供了用于注册结果、启动结果以及在系统分派结果后处理结果的组件。
所以没有必要再使用startActivityForResult
andonActivityResult
了。
为了使用新的 API,您需要在源 Activity 中创建一个 ActivityResultLauncher,指定目标 Activity 完成并返回所需数据时将运行的回调:
private val intentLauncher =
registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { result ->
if (result.resultCode == Activity.RESULT_OK) {
result.data?.getStringExtra("key1")
result.data?.getStringExtra("key2")
result.data?.getStringExtra("key3")
}
}
然后,在您需要时启动您的意图:
intentLauncher.launch(Intent(this, YourActivity::class.java))
要从目标 Activity 返回数据,您只需添加一个带有返回值的意图即可setResult()
:
val data = Intent()
data.putExtra("key1", "value1")
data.putExtra("key2", "value2")
data.putExtra("key3", "value3")
setResult(Activity.RESULT_OK, data)
finish()
有关任何其他信息,请参阅Android 文档
我创建了简单的演示类供您更好地参考。
第一活动.java
public class FirstActivity extends AppCompatActivity {
private static final String TAG = FirstActivity.class.getSimpleName();
private static final int REQUEST_CODE = 101;
private Button btnMoveToNextScreen;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btnMoveToNextScreen = (Button) findViewById(R.id.btnMoveToNext);
btnMoveToNextScreen.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent mIntent = new Intent(FirstActivity.this, SecondActivity.class);
startActivityForResult(mIntent, REQUEST_CODE);
}
});
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if(resultCode == RESULT_OK){
if(requestCode == REQUEST_CODE && data !=null) {
String strMessage = data.getStringExtra("keyName");
Log.i(TAG, "onActivityResult: message >>" + strMessage);
}
}
}
}
这是 SecondActivity.java
public class SecondActivity extends AppCompatActivity {
private static final String TAG = SecondActivity.class.getSimpleName();
private Button btnMoveToPrevious;
private EditText editText;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
editText = (EditText) findViewById(R.id.editText);
btnMoveToPrevious = (Button) findViewById(R.id.btnMoveToPrevious);
btnMoveToPrevious.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String message = editText.getEditableText().toString();
Intent mIntent = new Intent();
mIntent.putExtra("keyName", message);
setResult(RESULT_OK, mIntent);
finish();
}
});
}
}
在第一个活动中,您可以使用发送意图 startActivityForResult()
,然后在完成使用后从第二个活动中获取结果setResult
。
MainActivity.class
public class MainActivity extends AppCompatActivity {
private static final int SECOND_ACTIVITY_RESULT_CODE = 0;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
// "Go to Second Activity" button click
public void onButtonClick(View view) {
// Start the SecondActivity
Intent intent = new Intent(this, SecondActivity.class);
// send intent for result
startActivityForResult(intent, SECOND_ACTIVITY_RESULT_CODE);
}
// This method is called when the second activity finishes
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
// check that it is the SecondActivity with an OK result
if (requestCode == SECOND_ACTIVITY_RESULT_CODE) {
if (resultCode == RESULT_OK) {
// get String data from Intent
String returnString = data.getStringExtra("keyName");
// set text view with string
TextView textView = (TextView) findViewById(R.id.textView);
textView.setText(returnString);
}
}
}
}
SecondActivity.class
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
}
// "Send text back" button click
public void onButtonClick(View view) {
// get the text from the EditText
EditText editText = (EditText) findViewById(R.id.editText);
String stringToPassBack = editText.getText().toString();
// put the String to pass back into an Intent and close this activity
Intent intent = new Intent();
intent.putExtra("keyName", stringToPassBack);
setResult(RESULT_OK, intent);
finish();
}
}
所有这些答案都在解释您的第二个活动的场景需要在发送数据后完成。
但是,如果您不想完成第二个活动并希望将数据发送回第一个活动,那么您可以使用 BroadCastReceiver。
在第二个活动中 -
Intent intent = new Intent("data");
intent.putExtra("some_data", true);
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
在第一个活动中-
private BroadcastReceiver tempReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
// do some action
}
};
在 onCreate() 中注册接收者——
LocalBroadcastManager.getInstance(this).registerReceiver(tempReceiver,new IntentFilter("data"));
在 onDestroy() 中取消注册
根据您的情况,实现所需结果的另一种方法是创建一个侦听器接口。
通过让父活动侦听由子活动触发的接口,同时将所需数据作为参数传递,可以创建一组类似的情况
只是我认为上述答案中缺少的一个小细节。
如果您的子活动可以从多个父活动中打开,那么您可以setResult
根据您的活动是否由startActivity
或打开来检查您是否需要这样做startActivityForResult
。您可以通过使用getCallingActivity()
. 更多信息在这里。
有一些方法可以做到这一点。1.通过使用上面答案中很好解释的 startActivityForResult() 。
通过在您的“Utils”类或您自己的任何其他类中创建静态变量。例如,我想将 ActivityB 中的 studentId 传递给 ActivityA。首先我的 ActivityA 正在调用 ActivityB。然后在 ActivityB 中设置 studentId(它是 Utils.class 中的静态字段)。像这样 Utils.STUDENT_ID="1234"; 然后在返回 ActivityA 时使用存储在 Utils.STUDENT_ID 中的 studentId。
通过在应用程序类中创建 getter 和 setter 方法。
像这样:
public class MyApplication extends Application {
private static MyApplication instance = null;
private String studentId="";
public static MyApplication getInstance() {
return instance;
}
@Override
public void onCreate() {
super.onCreate();
instance = this;
}
public void setStudentId(String studentID){
this.studentId=studentID;
}
public String getStudentId(){
return this.studentId;
}
}
这样你就完成了。当你在 ActivityB 中时,只需在里面设置数据,然后回到 ActivityA ,获取数据。
使用sharedPreferences并保存您的数据并从应用程序的任何位置访问它
像这样保存日期
SharedPreferences sharedPreferences = getPreferences(MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putString(key, value);
editor.commit();
并接收这样的数据
SharedPreferences sharedPreferences = getPreferences(MODE_PRIVATE);
String savedPref = sharedPreferences.getString(key, "");
mOutputView.setText(savedPref);