前面两节所有应用都是同一个activity中的,是时候讲activity之间交互的操作了,此后会涉及到intent这个概念,这也算一个新的里程碑开始。
主要内容包括intent的使用,以及activity之间的数据传递。
假设有两个activity,MainActivity跟SecondActivity,其中MainActivity是主活动,现在MainActivity需要调用SecondActivity,则可以用以下两种方式。
1.activity跳转
(1)显示intent:使用显示intent,需要增添的配置只有调用方的activity
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_main); final Button btn_intent = (Button) findViewById(R.id.intent_btn); btn_intent.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(MainActivity.this,SecondActivity.class); startActivity(intent); } }); } ... }
如上述代码,通过点击btn_intent这个button,就可以调用SecondActivity。Intent(Context packageContext, Class<?> cls) 。这个构造函数接收两个参数,第一个参数 Context 要求提供一个启动活动的上下文,第二个 参数 Class 则是指定想要启动的目标活动,通过这个构造函数就可以构建出 Intent 的“意图”。 调用意图,使用startActivity即可。
(2)隐式intent
使用隐式intent可以不用指定activity,而只需要调用action name即可。
首先在AndroidManifest.xml中配置action跟category,这两者可以组成intent的过滤器
<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>
当MainActivity需要调用SecondActivity时,可以调用intent的另一个构造方法
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_main);
final Button btn_intent = (Button) findViewById(R.id.intent_btn);
btn_intent.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.example.activitytest.ACTION_START");
#如果category配置的是default,则此处可以不用加category
intent.addCategory("com.example.activitytest.MY_CATEGORY");
startActivity(intent);
}
});
}
...
}
ps:隐式调用更像是前端中常用的模式,通过给activity标示唯一识别符,然后需要用到时则通过唯一识别符来调用。而不用像显式调用那样,在编码时必须熟知调用的主从关系。
另外隐式调用的好处是隐式调用不仅可以启动自己程序内的活动,还可以启动其他程序的活动, 这使得 Android 多个应用程序之间的功能共享成为了可能。
(3)程序间的调用(隐式调用+)
sys_btn_intent.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); } });
其中ACTION_VIEW代表调用的是系统本身的activity
2.activity间数据传递
(1)数据传递
MainActivity中有一个字符串,现在想把这个字符串传递到SecondActivity,可以用putExtra跟getStringExtra在activity之间传递数据
MainActivity需要传递data给SecondActivity,具体使用如下:
MainActivity(sender):
button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { String data = " hello world!"; Intent intent = new Intent(MainActivity.this, SecondActivity.class); intent.putExtra("extra_data",data); //{"extra_data":data} startActivity(intent); } });
SecondActivity(receiver):
public class SecondActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState){ super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.second_layout); //通过intent获取数据 Intent intent = getIntent(); String data = intent.getStringExtra("extra_data"); Toast.makeText(SecondActivity.this,data,Toast.LENGTH_SHORT).show(); } }
(2)数据返回
因为数据返回,一般是activity的销毁,而不是调用,因此跟(1)中的方式不一样。假设现在需要从SecondActivity返回数据给MainActivity:
SecondActivity
public class SecondActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState){ super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); 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 MainActivity"); //绑定result_code跟intent内容 setResult(RESULT_OK,intent); finish(); } }); } }
MainActivity需要接受的参数有:一个是request_code,一个是result_code,最后一个是intent。
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_main); final Button button = (Button) findViewById(R.id.button_1); button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(MainActivity.this, SecondActivity.class); startActivityForResult(intent, 200); } }); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data){ switch (requestCode){ case 200: if (requestCode == RESULT_OK){ String returnedData = data.getStringExtra("data_return"); Log.d("MainActivity", returnedData); } break; default: } } ... }
3.在Bundle中传递跟保存数据
Bundle中传递跟保存数据是为了防止activity调用时,当前activity如果需要在onDestory时保存一些临时数据,则可以用onSaveInstanceState这个构造函数。
保存数据:
@Override protected void onSaveInstanceState(Bundle outState){ super.onSaveInstanceState(outState); String tempData = "Something you just typed"; outState.putString("data_key",tempData); }
取出数据:
public class ActivityLifeCycleTest extends AppCompatActivity { public static final String TAG = "MainActivity"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.d(TAG, "onCreate"); Toast.makeText(ActivityLifeCycleTest.this,TAG+"===onCreate==",Toast.LENGTH_SHORT).show(); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_activity_life_cycle_test); if (savedInstanceState != null){ String data = savedInstanceState.getString("data_key"); Log.d(TAG, data); } ... }