一. Activity
二. Service
Activity
官方文档:https://developer.android.google.cn/guide/components/activities
生命周期图:
创建第二个Activity
- 需要在清单文件中为其配置一个activity标签
-
标签中如果带有这个子节点,则会在系统中多创建一个快捷图标
-
<intent-filter>
-
<action android:name="android.intent.action.MAIN" />
-
<category android:name="android.intent.category.LAUNCHER" />
-
</intent-filter>
-
-
-
<application
-
android:allowBackup= "true"
-
android:icon= "@drawable/ic_launcher"
-
android:label= "@string/app_name"
-
android:theme= "@style/AppTheme" >
-
<activity
-
android:name= "com.itheima.createactivity.MainActivity"
-
android:icon= "@drawable/photo2"
-
android:label= "主界面" >
-
<intent-filter>
-
<action android:name="android.intent.action.MAIN" />
-
-
<category android:name="android.intent.category.LAUNCHER" />
-
</intent-filter>
-
</activity>
-
<activity android:name="com.itheima.createactivity.SecondActivity"
-
android:icon= "@drawable/photo3"
-
android:label= "第二个界面"
-
>
-
<intent-filter>
-
<action android:name="android.intent.action.MAIN" />
-
<category android:name="android.intent.category.LAUNCHER" />
-
</intent-filter>
-
</activity>
-
-
一个应用程序可以在桌面创建多个快捷图标。
-
activity的名称、图标可以和应用程序的名称、图标不相同
-
android:icon= "@drawable/ic_launcher"
-
android:label= "@string/app_name"
-
Activity的跳转
Activity的跳转需要创建Intent对象,通过设置intent对象的参数指定要跳转Activity
通过设置Activity的包名和类名实现跳转,称为显式意图
通过指定动作实现跳转,称为隐式意图
显式意图
-
跳转至同一项目下的另一个Activity,直接指定该Activity的字节码即可
-
Intent intent = new Intent();
-
intent.setClass( this, SecondActivity. class);
-
startActivity(intent);
-
-
跳转至其他应用中的Activity,需要指定该应用的包名和该Activity的类名
-
Intent intent = new Intent();
-
//启动系统自带的拨号器应用
-
intent.setClassName( "com.android.dialer", "com.android.dialer.DialtactsActivity");
-
startActivity(intent);
-
隐式意图
-
隐式意图跳转至指定Activity
-
Intent intent = new Intent();
-
//启动系统自带的拨号器应用
-
intent.setAction(Intent.ACTION_DIAL);
-
startActivity(intent);
-
-
要让一个Activity可以被隐式启动,需要在清单文件的activity节点中设置intent-filter子节点
-
<intent-filter >
-
<action android:name="com.itheima.second"/>
-
<data android:scheme="asd" android:mimeType="aa/bb"/>
-
<category android:name="android.intent.category.DEFAULT"/>
-
</intent-filter>
- action 指定动作(可以自定义,可以使用系统自带的)
- data 指定数据(操作什么内容)
- category 类别 (默认类别,机顶盒,车载电脑)
-
- 隐式意图启动Activity,需要为intent设置以上三个属性,且值必须与该Activity在清单文件中对三个属性的定义匹配
- intent-filter节点及其子节点都可以同时定义多个,隐式启动时只需与任意一个匹配即可
获取通过setData传递的数据
-
//获取启动此Activity的intent对象
-
Intent
intent
= getIntent();
-
Uri
uri
= intent.getData();
显式意图和隐式意图的应用场景
- 显式意图用于启动同一应用中的Activity
-
隐式意图用于启动不同应用中的Activity
- 如果系统中存在多个Activity的intent-filter同时与你的intent匹配,那么系统会显示一个对话框,列出所有匹配的Activity,由用户选择启动哪一个
实例代码:
-
<intent-filter >
-
<action android:name="com.itheima.sa2"/>
-
<data android:scheme="heima2" android:mimeType="text/username"/>
-
<category android:name="android.intent.category.DEFAULT"/>
-
</intent-filter>
-
</activity>
-
-
Java文件:
-
MainActivity. java:
-
-
/**
-
* 跳转至打电话activity
-
* 跳转至其他应用的activity
-
* 隐式跳转:通过指定action和data
-
* @param v
-
*/
-
public void click1( View v){
-
Intent intent = new Intent();
-
//隐式意图
-
intent. setAction( Intent. ACTION_CALL);
-
intent. setData( Uri. parse( "tel:110"));
-
//跳转
-
startActivity(intent);
-
}
-
-
/**
-
* 跳转至secondActivity
-
* 在本应用中跳转
-
* 显示跳转:直接指定目标Activity的包名和类名
-
* @param v
-
*/
-
public void click2( View v){
-
Intent intent = new Intent();
-
//cls:直接指定目标Activity的类名
-
//显示意图
-
intent. setClass( this, SecondActivity. class);
-
startActivity(intent);
-
}
-
-
/**
-
* 显示跳转至拨号器
-
*/
-
public void click3( View v){
-
Intent intent = new Intent();
-
//指定目标Activity的包名和类名
-
intent. setClassName( "com.android.dialer", "com.android.dialer.DialtactsActivity");
-
startActivity(intent);
-
}
-
/**
-
* 隐式跳转至拨号器
-
*/
-
public void click4( View v){
-
Intent intent = new Intent();
-
//隐式设置拨号器的动作
-
intent. setAction( Intent. ACTION_DIAL);
-
startActivity(intent);
-
}
-
-
/**
-
* 隐式跳转至secondActivity
-
* @param v
-
*/
-
public void click5( View v){
-
Intent intent = new Intent();
-
intent. setAction( "com.itheima.sa2");
-
//intent.setData(Uri.parse("heima2:qwe"));
-
//intent.setType("text/username");
-
//intent.setData(Uri.parse("heima2:qwe123"));
-
-
intent. setDataAndType( Uri. parse( "heima2:qwe123"), "text/username");
-
//系统会自动添加默认的category
-
intent. addCategory( Intent. CATEGORY_DEFAULT);
-
startActivity(intent);
-
}
-
-
/**
-
* 显式跳转至浏览器
-
*/
-
public void click6( View v){
-
Intent intent = new Intent();
-
intent. setClassName( "com.android.browser", "com.android.browser.BrowserActivity");
-
startActivity(intent);
-
}
-
/**
-
* 隐式跳转至浏览器
-
* @param v
-
*/
-
public void click7( View v){
-
Intent intent = new Intent();
-
intent. setAction( Intent. ACTION_VIEW);
-
intent. setData( Uri. parse( "http://www.baidu.com"));
-
startActivity(intent);
-
}
-
-
SecondActivity. java:
-
接收 MainActivity传递过来的数据:
-
Intent intent = getIntent();
-
Uri uri = intent. getData();
-
System. out. println(uri. toString());
-
清单文件:
-
<activity android:name=".SecondActivity">
-
<intent-filter >
<action android:name="com.itheima.sa"/>
-
<action android:name="com.itheima.sa3"/>
-
<data android:scheme="heima"/>
-
<data android:scheme="heima3"/>
-
<category android:name="android.intent.category.DEFAULT"/>
-
</intent-filter>
Activity跳转时的数据传递
-
Activity通过Intent启动时,可以通过Intent对象携带数据到目标Activity
-
Intent intent = new Intent( this, SecondActivity. class);
-
intent.putExtra( "maleName", maleName);
-
intent.putExtra( "femaleName", femaleName);
-
startActivity(intent);
-
-
在目标Activity中取出数据
-
Intent intent = getIntent();
-
String maleName = intent.getStringExtra( "maleName");
-
String femaleName = intent.getStringExtra( "femaleName");
-
-
通过Bundle传递数据
-
public void click (View v){
-
Intent intent = new Intent( this, SecondActivity.class);
-
//把数据封装至intent对象中
-
//intent.putExtra("malename", "李志");
-
//intent.putExtra("femalename", "芙蓉姐姐");
-
-
//把数据封装至bundle对象中
-
Bundle bundle = new Bundle();
-
bundle.putString( "malename", "李志");
-
bundle.putString( "femalename", "芙蓉姐姐");
-
//把bundle对象封装至intent对象中
-
intent.putExtras(bundle);
-
startActivity(intent);
-
}
-
-
接收数据:
-
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获取返回值
从A界面打开B界面, B界面关闭的时候,返回一个数据给A界面
步骤: 1. 开启activity并且获取返回值
startActivityForResult(intent, 0);
-
在新开启的界面里面实现设置数据的逻辑
-
Intent data = new Intent();
-
data.putExtra( "phone", phone);
-
//设置一个结果数据,数据会返回给调用者
-
setResult( 0, data);
-
finish(); //关闭掉当前的activity,才会返回数据
-
-
在开启者activity里面实现方法 onActivityResult(int requestCode, int resultCode, Intent data) 通过data获取返回的数据
- 根据请求码和结果码确定业务逻辑
Activity获取返回值
-
public
void
click
(View c){
-
//跳转至选择联系人Activity
-
Intent
intent
=
new
Intent(
this, ContactActivity.class);
-
//用这个api启动的Activity,在销毁时,系统会回调onActivityResult
-
startActivityForResult(intent,
10);
-
}
-
-
-
public
void
click2
(View v){
-
//跳转至选择快捷回复的Activity
-
Intent
intent
=
new
Intent(
this, CallbackActivity.class);
-
startActivityForResult(intent,
20);
-
}
-
//如果有Activity在销毁时返回了数据,那么就会调用此方法来接收数据
-
//requestCode:用来区分数据来自于哪一个Activity
-
//resultCode:用来区分返回的数据是什么类型的
-
@Override
-
protected
void
onActivityResult
(int requestCode, int resultCode, Intent data) {
-
// TODO Auto-generated method stub
-
super.onActivityResult(requestCode, resultCode, data);
-
-
String
name
= data.getStringExtra(
"name");
-
if(requestCode ==
10){
-
EditText
et
= (EditText)findViewById(R.id.et);
-
et.setText(name);
-
}
-
else
if(requestCode ==
20){
-
EditText
et_content
= (EditText)findViewById(R.id.et_content);
-
et_content.setText(name);
-
}
-
}
-
-
-
//Activity返回时传递数据,也是通过意图对象
-
Intent
data
=
new
Intent();
-
//把要传递的数据封装至意图对象中
-
data.putExtra(
"name", objects[position]);
-
//当前Activity销毁时,data这个意图就会传递给启动当前Activity的那个Activity
-
setResult(
1, data);
-
-
//销毁当前Activity
-
finish();
Activity生命周期
void onCreate()
- Activity已经被创建完毕
void onStart()
- Activity已经显示在屏幕,但没有得到焦点
void onResume()
- Activity得到焦点,可以与用户交互
void onPause()
- Activity失去焦点,无法再与用户交互,但依然可见
void onStop()
- Activity不可见,进入后台
void onDestroy()
- Activity被销毁
void onRestart()
- Activity从不可见变成可见时会执行此方法
前台Activity(有焦点)当执行finish()后,会一次执行onpause-->onStop-->onDestroy
启动应用到获取焦点:oncreat-->onstart-->onresume
前台Activity(有焦点)点击home键:onpause-->onstop,再次启动onrestart-->onstart-->onresume
使用场景
- Activity创建时需要初始化资源,销毁时需要释放资源;或者播放器应用,在界面进入后台时需要自动暂停
完整生命周期(entire lifetime)
onCreate-->onStart-->onResume-->onPause-->onStop-->onDestory
可视生命周期(visible lifetime)
onStart-->onResume-->onPause-->onStop
前台生命周期(foreground lifetime)
onResume-->onPause
Activity的四种启动模式
每个应用会有一个Activity任务栈,存放已启动的Activity
Activity的启动模式,修改任务栈的排列情况
- standard 标准启动模式
- singleTop 单一顶部模式
- 如果任务栈的栈顶存在这个要开启的activity,不会重新的创建activity,而是复用已经存在的activity。保证栈顶如果存在,不会重复创建。
- 应用场景:浏览器的书签
-
singeTask 单一任务栈,在当前任务栈里面只能有一个实例存在
- 当开启activity的时候,就去检查在任务栈里面是否有实例已经存在,如果有实例存在就复用这个已经存在的activity,并且把这个activity上面的所有的别的activity都清空,复用这个已经存在的activity。保证整个任务栈里面只有一个实例存在
- 应用场景:浏览器的activity
- 如果一个activity的创建需要占用大量的系统资源(cpu,内存)一般配置这个activity为singletask的启动模式。webkit内核 c代码
-
singleInstance启动模式非常特殊, activity会运行在自己的任务栈里面,并且这个任务栈里面只有一个实例存在
- 如果你要保证一个activity在整个手机操作系统里面只有一个实例存在,使用singleInstance
- 应用场景: 电话拨打界面
横竖屏切换的生命周期
默认情况下 ,横竖屏切换, 销毁当前的activity,重新创建一个新的activity 快捷键ctrl+F11
在一些特殊的应用程序常见下,比如游戏,不希望横竖屏切换activity被销毁重新创建 需求:禁用掉横竖屏切换的生命周期 1. 横竖屏写死 android:screenOrientation="landscape" android:screenOrientation="portrait"
-
让系统的环境 不再去敏感横竖屏的切换。
android:configChanges="orientation|screenSize|keyboardHidden"
-
设置当前Activity的方向,使用代码控制横竖屏
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
常用方法:
-
删除title,在onCreat方法调用 requestWindowFeature(Window.FEATURENOTITLE);