Activity中文翻译过来是"活动"的意思,是Android四大组件中唯一一个可以直观感受到的组件,也就是可以看到的,它的主要作用是展示界面和用户交互,下面就Activity的一些基本知识进行探究:
1.一个Activity的基本组成.
在用Android studio新建一个Activity,它会自动帮你构建一个继承Activity的java类,一个layout文件,layout文件用我们用来定义的界面,java类用来显示我们定义的layout文件,当然为了能让这个Activity显示出来.还需要在AndroidManifest.xml文件中配置,比如我们创建了一个ActivityA,需要在AndroidManifest.xml中配置.
<activity android:name=".ActivityA"/>
2.Activity的基本使用:
修改界面title内容:在Activity类中调用setTitle()方法,或者在AndroidManifest.xml中配置Activity组件的时候加上android:label,如果既没有调用setTitle方法,也没有添加android:label标签,那么就是application节点下的android:label标签,在application节点下的这个标签同时也是launcher界面app的名称.
消息提醒使用:
mButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(MainActivity.this, " add ", Toast.LENGTH_SHORT).show();
ActivityA.start(MainActivity.this);
}
});
添加menu:
在res文件夹中新建一个menu文件夹,新建内容:
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<!--同样可以设置图片 也可以设置显示状态(隐藏 显示 空间不足时隐藏)默认是隐藏的-->
<item android:id="@+id/add"
android:title="Add"/>
<item android:title="Remove"
android:id="@+id/remove"/>
</menu>
在Activity中重写如下两个方法:
@Override
//加载布局 返回值是true是表示显示 false表示不显示
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.main_menu, menu);
return true;
}
//响应事件
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.add:
Toast.makeText(this, " add ", Toast.LENGTH_SHORT).show();
break;
case R.id.remove:
Toast.makeText(this, " remove ", Toast.LENGTH_SHORT).show();
break;
default:
}
return false;
}
销毁Activity:
finish();//销毁活动
isFinishing()//是否正在销毁
除了生命周期方法之外其他重要的回调方法(可以结合生命周期方法研究activity更加精细的生命周期):
@Override
//activity附着到window上
public void onAttachedToWindow() {
super.onAttachedToWindow();
Log.d(TAG, "onAttachedToWindow: ");
}
@Override
//activity从window上剥离
public void onDetachedFromWindow() {
super.onDetachedFromWindow();
Log.d(TAG, "onDetachedFromWindow: ");
}
@Override
//按back键的调用时机 禁掉super方法的调用,按back键将不能退出activity.
public void onBackPressed() {
super.onBackPressed();
Log.d(TAG, "onBackPressed: ");
}
@Override
//layout资源加载到window的调用时机
public void onContentChanged() {
super.onContentChanged();
Log.d(TAG, "onContentChanged: ");
}
@Override
//布局绘制完成 activity完全可见
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
Log.d(TAG, "onWindowFocusChanged: " + " focus " + hasFocus);
}
3.Activity的两种启动方式:
两种启动方式主要在构造intent对象的方式上不同.
显示启动(上篇文章中讲的Live Templates中有默认快捷设置):
构造intent时需要传递两个参数,当前activity对象,待跳转的class对象.在被启动的Activity中定义一个静态方法,如果跳转时需要携带数据,可以增加这个方法的参数个数.
public static void start(Context context) {
Intent starter = new Intent(context, ActivityA.class);
context.startActivity(starter);
}
隐式启动:
在AndroidManifest.xml中注册组件主要会有三部分(action category和data)内容,在代码中也要设置对应的三部分才能精确匹配上.一般在AndroidManifest.xml中注册如下:
<activity
android:name=".ActivityB" >
<intent-filter>
<action android:name="com.dwj.test"/>
//必须设置这个category,否则不能启动这个activity
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
在代码中启动ActivityB的方式如下:
Intent intent = new Intent("com.dwj.test");
//对应的category系统默认帮你加上了
startActivity(intent);
当然可以设置多个action和多个category,比如设置成:
<activity
android:name=".ActivityB" >
<intent-filter>
<action android:name="com.dwj.test"/>
<action android:name="com.dwj.test1"/>
<category android:name="android.intent.category.DEFAULT"/>
<category android:name="com.dwj.category"/>
</intent-filter>
</activity>
这里就需要注意在代码中intent设置的action和category时的匹配规则:
action:设置其中任何一个都可以,当然你也可以设置两个action
category:在AndroidManifest中使用隐式启动的activity一定要设置android.intent.category.DEFAULT这个flag,否则会启动失败,系统会默认设置这个category,如果像上面这样设置多个category,在代码中你可以加上除了android.intent.category.DEFAULT其他的flag,也可以不加上,activity还是可以被启动成功的.
当然除了android.intent.category.DEFAULT这个flag用来设置activity隐式启动之外,还有一个特殊的category的flag.
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
app的主Activity,是从launcher图标跳转到app界面,这个是固定设置,是安装时给系统解析设置app主界面的.
一般设置隐式启动时设置上面两项就可以了,在启动一些特殊activity时还需要配置data信息,data配置可以看成两部分URI 和mimeType,具体说明如下:
android:scheme :指定数据协议部分,比如网页请求"http",电话请求"tel:",地理位置"geo:"
android:host :指定数据的主机名部分.
android:port :指定数据端口部分.
android:path:指定主机名和端口之后的部分
android:mimeType :处理的数据类型
uri的通用格式如下:
<scheme>://<host>:<port>/[<path>|<pathPrefix>|<pathPattern>]
比如像浏览器发一个网页请求:
Uri webpage = Uri.parse("http://www.android.com");
Intent webIntent = new Intent(Intent.ACTION_VIEW, webpage);
可以看到对应的uri结构.在代码中构造intent的时候,需要设置在AndroidManifest中设置的所有intent-filter信息.另外在设置data和Type是需要调用setDataAndType,不能先设置data在设置type,因为在源码中设置方法都会先把对方的数据清除掉:
public Intent setType(String type) {
mData = null;
mType = type;
return this;
}
更多关于intent的隐式跳转的例子可以参考官方文档:
(https://developer.android.google.cn/training/basics/intents/sending.html)
当然为了程序的健壮性,可以在隐式跳转前判断需要跳转的activity是否存在,用如下方式判断:
PackageManager packageManager = getPackageManager()
;List activities = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);boolean isIntentSafe = activities.size() > 0;
也可以用packageManager的resolveActivity方法,在packageManager同样有用来检查其他三大组件是否存在的方法.