Android页面跳转和数据回传
今天我尝试用两个页面实现数据的传递和回传,出现了一些问题,把问题已经成功的案例总结如下:
具体是这样的:
有两个layout布局,两个activity。MainActivity对应的布局里有一个文本框,给文本框添加点击事件,当点击文本框时,会跳转到MoreActivity中并显示另一个布局界面,
另一个布局界面里若干条数据(使用了checkBox)用户自由选择之后,点击确定按钮,可以跳转回到MainActivity对应的界面。同时这个界面里面的文本框内容就是用
户自己选择的文本内容。
刚开始我想使用bundle实现传值,如果使用这种方式,把这一部分代码写在onCreate()里面,那么只要打开主页面方法就会被调用,可此时bundle并没有接收到来自另
一个页面传来的值,所以就会抛出异常(NullPointerException)这说明把跳转和数据接收写在同一个位置是不可行的,那么就需要把页面跳转和接收回传的数据这两个
功能分开写。
如果想在Activity中得到新打开Activity 关闭后返回的数据,需要使用系统提供的startActivityForResult(Intent intent, int requestCode)方法打开新的Activity,
新的Activity 关闭后会向前面的Activity传回数据,为了得到传回的数据,必须在前面的Activity中重写onActivityResult(int requestCode, int resultCode, Intent data)方法。
注意:startActivityForResult这个方法里会有requestCode参数,这个参数是要告诉即将跳转到的界面,跳转前的界面的“名字代号”就是requestCode,那么相应地,在跳转
到的界面如果要返回,返回的时候同样会带上一个resultCode,这个是确定这是从哪个界面跳回去的,如果只有两个页面,那这些码也就不重要了,如果要实现复杂的页
面跳转功能,就必须使用这两个参数来确定跳转唯一性。
这是系统定义的参数码的常量值:
请求码作用举例:如果A页面有两个按钮,分别实现的是不同的功能,但是点击这两个按钮都会跳转到B页面,B页面处理之后也会跳回A页面,但是B页面处理地是不同的
按钮,这时候就需要A页面的两个按钮在发出请求的时候就给两个请求带上不同的requestCode,那么B页面处理完之后也要回传不同的resultCode这样两个不同的按钮和两
组不同的数据就不会再发生冲突,在A页面重写onActivityResult(int requestCode, int resultCode, Intent data)方法,步骤是这样的,首先确定requestCode使用if...else语句或
者switch...case语句,然后再区分不同的resultCode,最后再接收回传的数据结果。
好了,现在上代码:
activity_main.xml
1 <EditText 2 android:id="@+id/tv_text5" 3 android:layout_width="300dp" 4 android:layout_height="154dp" 5 android:layout_above="@+id/tv_button1" 6 android:layout_alignParentEnd="true" 7 android:layout_alignParentRight="true" 8 android:layout_marginEnd="11dp" 9 android:layout_marginRight="11dp" 10 android:layout_marginBottom="6dp" 11 android:hint="点击跳转填写" 12 android:onClick="onClick" 13 android:textColor="@android:color/holo_purple" />
activity_more.xml
1 <?xml version="1.0" encoding="utf-8"?> 2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 3 android:layout_width="match_parent" 4 android:layout_height="match_parent" 5 android:orientation="vertical" 6 > 7 8 <TextView 9 android:id="@+id/tvHead" 10 android:layout_width="wrap_content" 11 android:layout_height="wrap_content" 12 android:layout_centerHorizontal="true" 13 android:text="特殊情况" 14 android:layout_gravity="center" 15 android:textColor="@color/red" 16 android:background="@color/colorPrimary" 17 android:textSize="30dp" /> 18 <TextView 19 android:layout_width="wrap_content" 20 android:layout_height="wrap_content" 21 android:textColor="@color/coral" 22 android:text="多选:" 23 android:textSize="23dp" 24 android:layout_marginLeft="40dp" 25 /> 26 <CheckBox 27 android:id="@+id/cb_0" 28 android:layout_width="wrap_content" 29 android:layout_height="wrap_content" 30 android:layout_below="@+id/tvHead" 31 android:layout_marginTop="32dp" 32 android:layout_marginLeft="40dp" 33 android:layout_toLeftOf="@+id/tvHead" 34 android:textSize="20dp" 35 android:text="无" /> 36 37 <CheckBox 38 android:id="@+id/cb_1" 39 android:layout_width="wrap_content" 40 android:layout_height="wrap_content" 41 android:layout_alignBaseline="@+id/cb_2" 42 android:layout_alignBottom="@+id/cb_2" 43 android:layout_marginLeft="40dp" 44 android:layout_toRightOf="@+id/cb_2" 45 android:textSize="20dp" 46 android:text="2021年2月1日至今去过或者现在居住在中高风险地区" /> 47 <CheckBox 48 android:id="@+id/cb_2" 49 android:layout_width="wrap_content" 50 android:layout_height="wrap_content" 51 android:layout_alignBaseline="@+id/cb_0" 52 android:layout_alignBottom="@+id/cb_0" 53 android:layout_alignLeft="@+id/cb_0" 54 android:layout_marginLeft="40dp" 55 android:textSize="20dp" 56 android:text="本人或家人正在集中隔离" /> 57 <CheckBox 58 android:id="@+id/cb_3" 59 android:layout_width="wrap_content" 60 android:layout_height="wrap_content" 61 android:layout_alignBaseline="@+id/cb_0" 62 android:layout_alignBottom="@+id/cb_0" 63 android:layout_alignLeft="@+id/cb_0" 64 android:layout_marginLeft="40dp" 65 android:textSize="20dp" 66 android:text="本人或家人有出现发热症状" /> 67 <CheckBox 68 android:id="@+id/cb_4" 69 android:layout_width="wrap_content" 70 android:layout_height="wrap_content" 71 android:layout_alignBaseline="@+id/cb_0" 72 android:layout_alignBottom="@+id/cb_0" 73 android:layout_alignLeft="@+id/cb_0" 74 android:layout_marginLeft="40dp" 75 android:textSize="20dp" 76 android:text="居住地改变" /> 77 78 <Button 79 android:id="@+id/btn_add" 80 android:layout_width="66dp" 81 android:layout_height="38dp" 82 android:layout_below="@+id/cbSing" 83 android:layout_centerHorizontal="true" 84 android:layout_marginLeft="260dp" 85 android:layout_marginTop="36dp" 86 android:onClick="onClick" 87 android:text="确定" 88 android:textSize="10dp" /> 89 </LinearLayout>
MainActivity.java
1 private TextView text5;
2 text5=findViewById(R.id.tv_text5);
3 //点击跳转事件
4 public void onClick(View view) {
5 if (view.getId() == R.id.tv_text5) {
6 Intent intent = new Intent(this, MoreActivity.class);
7 intent.putExtra("flag",RESQUEST);
8 startActivityForResult(intent,RESQUEST);
9 }
10 }
11
12 //接收复选框传过来的数据
13 @Override
14 protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
15 super.onActivityResult(requestCode, resultCode, data);
16 switch (resultCode)
17 {
18 case RESULT_OK:
19 tempstr=data.getExtras().getString("mess");
20 Log.v("输出“特殊情况”获取结果 ",data.getExtras().getString("mess"));
21 text5.setText(tempstr);
22 break;
23 default:
24 break;
25 }
26 }
MoreActivity.java
1 package com.example.helloworld;
2
3 import android.content.Intent;
4 import android.os.Bundle;
5 import android.util.Log;
6 import android.view.View;
7 import android.widget.Button;
8 import android.widget.CheckBox;
9 import android.widget.CompoundButton;
10 import android.widget.EditText;
11 import android.widget.TextView;
12
13 import androidx.annotation.Nullable;
14 import androidx.appcompat.app.AppCompatActivity;
15
16 public class MoreActivity extends AppCompatActivity implements View.OnClickListener {
17 private CheckBox cb_0;
18 private CheckBox cb_1;
19 private CheckBox cb_2;
20 private CheckBox cb_3;
21 private CheckBox cb_4;
22 String str1="";
23 String str2="";
24 String str3="";
25 String str4="";
26 String str0="";
27 String str="";
28
29 @Override
30 protected void onCreate(@Nullable Bundle savedInstanceState) {
31 super.onCreate(savedInstanceState);
32 setContentView(R.layout.activity_more);
33 //初始化多选功能的按钮
34 initMore();
35 //设置监听器
36 setListener();
37 //设置监听器
38 // 初始化控件对象
39 Button btn_add = findViewById(R.id.btn_add);
40 // 绑定点击监听器
41 btn_add.setOnClickListener(this);
42 }
43
44 public void onClick(View view) {
45 if (view.getId() == R.id.btn_add) {
46 Intent intent = new Intent();
47 intent.putExtra("mess",str);
48 Log.v("MoreActivity得到的结果: ",str);
49 setResult(RESULT_OK,intent);
50 this.finish();
51 }
52 }
53 CompoundButton.OnCheckedChangeListener myCheckChangelistener = new CompoundButton.OnCheckedChangeListener() {
54 @Override
55 public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
56 // TODO Auto-generated method stub
57 //设置TextView的内容显示CheckBox的选择结果
58 setText();
59 }
60 };
61 //把多选框选中的东西放到主界面的text5上面,
62 // 并且把选中的记录放到数据库。
63 private void setText()
64 {
65
66 if(cb_0.isChecked())
67 {
68 str0 = cb_0.getText().toString();
69 cb_1.setChecked(false);
70 cb_2.setChecked(false);
71 cb_3.setChecked(false);
72 cb_4.setChecked(false);
73 // Log.v("cb_0: ",str0);
74 }
75 else{
76 str0="";
77 }
78 if(cb_0.isChecked()==false)
79 {
80 if(cb_1.isChecked())
81 {
82 str1= cb_1.getText().toString();
83 // Log.v("cb_1: ",str1);
84 }
85 else{
86 str1="";
87 }
88 if(cb_2.isChecked())
89 {
90 str2 = cb_2.getText().toString();
91 // Log.v("cb_2: ",str2);
92 }
93 else{
94 str2="";
95 }
96 if(cb_3.isChecked())
97 {
98 str3 = cb_3.getText().toString();
99 // Log.v("cb_3: ",str3);
100 }
101 else{
102 str3="";
103 }
104 if(cb_4.isChecked())
105 {
106 str4 = cb_4.getText().toString();
107 // Log.v("cb_4: ",str4);
108 }
109 else{
110 str4="";
111 }
112 }
113 str=str0+" "+str1+" "+str2+" "+str3+" "+str4;
114 Log.v("choicesResult: ",str);
115 }
116 private void initMore(){
117 cb_0=findViewById(R.id.cb_0);
118 cb_1=findViewById(R.id.cb_1);
119 cb_2=findViewById(R.id.cb_2);
120 cb_3=findViewById(R.id.cb_3);
121 cb_4=findViewById(R.id.cb_4);
122 }
123 //
124 //给多选框设置监听器
125
126 private void setListener(){
127 cb_0.setOnCheckedChangeListener(myCheckChangelistener);
128 cb_1.setOnCheckedChangeListener(myCheckChangelistener);
129 cb_2.setOnCheckedChangeListener(myCheckChangelistener);
130 cb_3.setOnCheckedChangeListener(myCheckChangelistener);
131 cb_4.setOnCheckedChangeListener(myCheckChangelistener);
132 }
133 }
在这一部分代码里,我出现了一个小问题,就是在跳转的时候会有一个finish()函数的调用,如果调用不当就会出现闪退的情况。
finish()
用于结束一个Activity的生命周期
问题是这样的:假如从A跳转到B,那么A里面是不可以写这个方法的,因为一个页面不可以既跳转,又结束自身声明周期。B页面里需要回跳的时候要写这个方法。
不使用finish():从activity 1中启动activity 2,然后在activity 2 启动activity 3, 这时按下返回键 程序就返回到了activity 2,再按下返回键 就返回到activity 1;
使用finish():从activity 1中启动activity 2,在activity 2调用finish(),然后在activity 2 启动activity 3,这时按下返回键 程序就直接返回了activity 1
注意:
如果在startActivityForResult起来的Activity里面设置setResult,
结果并不会马上返回给parent的Activity,只有当前Activity被finish,
结果才会被发送给parent的onActivityResult去处理!
那么也就是说,被跳转的页面必须写finish()方法,数据才可以完成回传。
activity返回result是在被finish的时候,也就是说调用setResult()方法必须在finish()之前。
那么如果在如下方法中调用setResult()也有可能不会返回成功:
onPause(), onStop(), onDestroy(),
因为这些方法调用不一定是在finish之前的。
Tomorrow the birds will sing.