安卓四大组件之一
创建Activity
- 定义java类,继承Activity
- 在清单文件中配置activity标签
多个入口
- 在桌面有多个应用,一个应用对应一个入口
- 一个应用程序可以在桌面创建多个快捷图标。
- activity的名称、图标可以和应用程序的名称、图标不相同
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.xxx.createactivity.MainActivity"
android:label="Main"
android:icon="@drawable/photo2"
>
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:icon="@drawable/photo3"
android:label="Second"
android:name="com.xxx.createactivity.SecondActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
跳转
显式跳转
- 同一应用中,指定目标Activity的字节码和当前的上下文
- 不同应用中,指定目标Activity所在的应用的包名和目标Activity的包名加类名
Intent intent = new Intent();
intent.setClass(this, SecondActivity.class);
startActivity(intent);
Intent intent = new Intent();
//启动系统自带的拨号器应用
intent.setClassName("com.android.dialer", "com.android.dialer.DialtactsActivity");
startActivity(intent);
隐式跳转
- 系统会在所有清单文件中寻找与程序员创建的intent匹配的intent-filter,找到则启动,找不到则抛异常
- 匹配就是intent-filter中定义了什么属性,程序员创建的intent中也必须设置什么属性
- 在清单文件的activity标签下,配置intent-filter子节点,其中再配置action和category
- action 指定动作(可以自定义,可以使用系统自带的)
- data 指定数据(操作什么内容)
- category 类别 (默认类别,机顶盒,车载电脑)
- 隐式意图启动Activity,需要为intent设置以上三个属性,且值必须与该Activity在清单文件中对三个属性的定义匹配
- intent-filter节点及其子节点都可以同时定义多个,隐式启动时只需与任意一个匹配即可
<intent-filter >
<action android:name="com.xxx.second"/>//必须有,可以有多个(一般只有一个),匹配一个即可
<data android:scheme="asd" android:mimeType="text/name"/>//可有可无,可以有多个,匹配一个即可,传递不同的数据
<category android:name="android.intent.category.DEFAULT"/>//必须有
</intent-filter>
public void click5(View v){
Intent intent = new Intent();
intent.setAction("com.xxx.second");
//匹配mimetype
// intent.setType("text/name"); //会清除data
//匹配scheme
// intent.setData(Uri.parse("asd:heooooo")); //会清除mimetype
intent.setDataAndType(Uri.parse("asd:heooooo"), "text/name");
//如果没有设置Category的匹配,那么自动匹配CATEGORY_DEFAULT
// intent.addCategory(Intent.CATEGORY_DEFAULT);
startActivity(intent);
}
//获取通过setData传递的数据
//获取启动此Activity的intent对象
Intent intent = getIntent();
Uri uri = intent.getData();
应用场景
- 启动同一应用中的Activity,用显式
- 启动不同应用中的Activity,用隐式
- 显式启动效率高于隐式
- 如果系统找到了多个intent-filter与程序员创建的intent匹配,那么就会弹出对话框,列举所有匹配的Activity,让用户选择
数据传递
- Activity跳转时,可以把数据封装在intent对象中
- intent中可以封装的数据类型:八大基本数据类型和字符串及它们的数组,还有实现了序列化接口的对象,还有bundle对象
- 数据可以先封装至Bundle,再把Bundle封装至intent
//Activity通过Intent启动时,可以通过Intent对象携带数据到目标Activity
Intent intent = new Intent(this, SecondActivity.class);
//intent.putExtra("maleName", maleName);
//intent.putExtra("femaleName", femaleName);
//Bundle中也可以用键值对封装数据,封装类型与intent一致
Bundle extras = new Bundle();
extras.putString("maleName", maleName);
extras.putString("femaleName", femaleName);
intent.putExtras(extras);
startActivity(intent);
//在目标Activity中取出数据
Intent intent = getIntent();
//String maleName = intent.getStringExtra("maleName");
//String femaleName = intent.getStringExtra("femaleName");
Bundle bundle = intent.getExtras();
String maleName = bundle.getString("maleName");
String femaleName = bundle.getString("femaleName");
横竖屏切换
- 默认情况横竖屏切换会触发生命周期方法重新执行,Activity销毁重建
用以下代码让横竖屏切换时不重建Activity
android:configChanges="orientation|screenSize|keyboardHidden"
用以下代码写死屏幕方向
android:screenOrientation="portrait"
生命周期
void onCreate()
- Activity已经被创建完毕
void onStart()
- Activity已经显示在屏幕,但没有得到焦点
void onResume()
- Activity得到焦点,可以与用户交互
void onPause()
- Activity失去焦点,无法再与用户交互,但依然可见
void onStop()
- Activity不可见,进入后台
void onDestroy()
- Activity被销毁, 按返回键会执行到这,但进程还在
void onRestart()
- Activity从不可见变成可见时会执行此方法
使用场景
- Activity创建时需要初始化资源,销毁时需要释放资源;或者播放器应用,在界面进入后台时需要自动暂停
完整生命周期(entire lifetime)
onCreate–>onStart–>onResume–>onPause–>onStop–>onDestory
可视生命周期(visible lifetime)
onStart–>onResume–>onPause–>onStop
前台生命周期(foreground lifetime)
onResume–>onPause
四种启动模式
每个应用会有一个Activity任务栈,存放已启动的Activity
Activity的启动模式,修改任务栈的排列情况
- standard 标准启动模式
- singleTop 单一顶部模式
- 如果任务栈的栈顶存在这个要开启的activity,不会重新的创建activity,而是复用已经存在的activity。保证栈顶如果存在,不会重复创建。
- 应用场景:浏览器的书签
singeTask 单一任务栈,在当前任务栈里面只能有一个实例存在
- 当开启activity的时候,就去检查在任务栈里面是否有实例已经存在,如果有实例存在就复用这个已经存在的activity,并且把这个activity上面的所有的别的activity都清空,复用这个已经存在的activity。保证整个任务栈里面只有一个实例存在
- 应用场景:浏览器的activity
- 如果一个activity的创建需要占用大量的系统资源(cpu,内存)一般配置这个activity为singletask的启动模式。webkit内核 c代码
singleInstance启动模式非常特殊, activity会运行在自己的任务栈里面,并且这个任务栈里面只有一个实例存在(跳转到其他activity时 切换到后台,切换回来时 显示在前台)
- 如果你要保证一个activity在整个手机操作系统里面只有一个实例存在,使用singleInstance
- 应用场景: 电话拨打界面
横竖屏切换的生命周期
默认情况下 ,横竖屏切换, 销毁当前的activity,重新创建一个新的activity
快捷键ctrl+F11
在一些特殊的应用程序常见下,比如游戏,不希望横竖屏切换activity被销毁重新创建
需求:禁用掉横竖屏切换的生命周期
横竖屏写死
android:screenOrientation=”landscape”
android:screenOrientation=”portrait”让系统的环境 不再去敏感横竖屏的切换。
android:configChanges="orientation|screenSize|keyboardHidden"
开启activity并获取返回值
从A界面打开B界面, B界面关闭的时候,返回一个数据给A界面
步骤:
开启activity并且获取返回值
startActivityForResult(intent, 0);
在新开启的界面里面实现设置数据的逻辑
Intent data = new Intent(); data.putExtra("phone", phone); //设置一个结果数据,数据会返回给调用者 setResult(0, data); finish();//关闭掉当前的activity,才会返回数据
在开启者activity里面实现方法
//通过data获取返回的数据 onActivityResult(int requestCode, int resultCode, Intent data) { }
- 通过判断请求码和结果码确定返回值的作用
public class MainActivity extends Activity {
public static final int CONTACT = 10;
public static final int SMS = 20;
public static final int HANDLE_EXCEPTION = 30;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click(View v){
Intent intent = new Intent(this, ContactActivity.class);
//启动一个Activity去获取数据(结果)
startActivityForResult(intent, CONTACT);
}
public void click2(View v){
Intent intent = new Intent(this, SmsActivity.class);
startActivityForResult(intent, SMS);
}
public void click3(View v){
Intent intent = new Intent(this, HandleExceptionActivity.class);
startActivityForResult(intent, HANDLE_EXCEPTION);
}
//只有通过startActivityForResult启动的Activity销毁时,才会回调这个方法,方法中传入的intent就封装了返回的数据
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
//通过请求码来判断data来自哪一个Activity
if(requestCode == CONTACT){
String name = data.getStringExtra("name");
EditText et_name = (EditText) findViewById(R.id.et_name);
et_name.setText(name);
}
else if(requestCode == SMS){
String sms = data.getStringExtra("sms");
EditText et_content = (EditText) findViewById(R.id.et_content);
et_content.setText(sms);
}
//先通过请求码判断data来自哪个Activity
else if(requestCode == HANDLE_EXCEPTION){
//通过结果码判断数据data中的数据是什么类型
if(resultCode == 100){
String name = data.getStringExtra("name");
EditText et_name = (EditText) findViewById(R.id.et_name);
et_name.setText(name);
}
else if(resultCode == 200){
String sms = data.getStringExtra("sms");
EditText et_content = (EditText) findViewById(R.id.et_content);
et_content.setText(sms);
}
}
}
}
public class ContactActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_contact);
final String[] names = new String[]{
"春晓",
"苗润",
"爱的结晶"
};
//显示联系人列表
ListView lv = (ListView) findViewById(R.id.lv);
lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv, names));
lv.setOnItemClickListener(new OnItemClickListener() {
//position:点击某个条目时,会通过position告诉程序员点击的是哪一个条目
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
Intent data = new Intent();
//把要传递的数据封装至intent中
data.putExtra("name", names[position]);
//当此Activity销毁时,返回至上一个Activity时,会把这个intent对象传递给上一个Activity
setResult(10, data);
//销毁当前Activity
finish();
}
});
}
}
public class HandleExceptionActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_handleexception);
final TextView tv_name = (TextView) findViewById(R.id.tv_name);
final TextView tv_content = (TextView) findViewById(R.id.tv_content);
tv_name.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent data = new Intent();
data.putExtra("name", tv_name.getText());
setResult(100, data);
finish();
}
});
tv_content.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent data = new Intent();
data.putExtra("sms", tv_content.getText());
setResult(200, data);
finish();
}
});
}
}
public class SmsActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_contact);
final String[] smss = new String[]{
"我正在开会,一会回你",
"我正在7饭,一会回你",
"我已死,有事烧纸"
};
//显示联系人列表
ListView lv = (ListView) findViewById(R.id.lv);
lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv, smss));
lv.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
Intent data = new Intent();
data.putExtra("sms", smss[position]);
setResult(20, data);
finish();
}
});
}
}
finish()只是通知要销毁activity,后面的代码还是会走完的,例如activity切换时动画的代码必须写在finish()后