Android

Android四大组件之Activity

2019-03-09  本文已影响35人  12313凯皇

Activity是一种展示型组件,也是Android四大组件中唯一一个用户能够直接感知到的。所以对用户来说Activity就是一个Android应用的全部,因此Activity的重要性不言而喻,那么下面开始接入正题:

一、生命周期

可以说Activity的生命周期是面试中必考的基础知识。Activity的整个生命周期涉及到七大方法:onCreate()onStart()onResume()onPause()onStop()onDestory()onRestart()
下面将详细介绍一下这个七个方法:

这样拆开来讲可能会有点抽象不易理解和记忆,所以我们可以通过两两组合来记或许会更容易一些:

下面再放上一张图片辅助理解和记忆:


推荐阅读:Activity生命周期之我见,上面那张图就源于此文章,另外这篇文章中还举了一个较为形象的例子:我们把一个Activity比作一本书,那么如果我现在要看一本书A,我需要先从书架取出这本书(onCreate),然后放到书桌上(onStart),接着翻开书(onResume),这时我们就可以开始看了。如果这时我们突然想去看书B,那我们就需要先合上书A或者直接走到书架旁(书A的onPause),然后同样的取出书B(书B的onCreate),将书放到书桌上(书B的onStart),然后翻开书(书B的onResume),如果此时书B完全遮盖住了书A的话,那么书A的onStop方法就会执行,如果没有完全遮盖住则不会调用。

如果这样还是不能很好的理解的话,下面再举出一些例子来强化记忆:

二、异常情况下的生命周期分析

什么叫异常情况呢,例如手机内存不足了,那么系统后台可能就会自行销毁一些目前没有在用的Activity。此时就设置到另外两个新方法了:onSaveInstanceState()onRestoreInstanceState()。这两个方法并不是生命周期方法,所以就并不一定触发,当应用遇到意外情况(如:内存不足、用户直接按Home键)由系统销毁一个Activity时,onSaveInstanceState() 会被调用。但是当用户主动去销毁一个Activity时,例如在应用中按返回键,onSaveInstanceState()就不会被调用。因为在这种情况下,用户的行为决定了不需要保存Activity的状态。通常onSaveInstanceState()只适合用于保存一些临时性的状态,而onPause()适合用于数据的持久化保存


推荐阅读:onSaveInstanceState和onRestoreInstanceState详解Activity详解(二)——异常情况下的生命周期分析

这里有一个相对较为常见的实例:当手机横竖屏切换时。此时会依次调用onPause()onSaveInstanceState(Bundle outState)onStop()onDestory()onCreate()onStart()onRestoreInstanceState(Bundle savedInstanceState)以及onResume()方法。

注意onSaveInstanceState(Bundle outState)调用时机在onStop()之前,但和onPause()没有既定的时序关系,即它既可能在onPause()之前调用,也可能在其之后调用。同样的,onRestoreInstanceState(Bundle savedInstanceState)的调用时机在onStart()之后。

三、启动模式

Activity总共有四种启动模式(LaunchMode):

推荐阅读:Activity的四种LaunchMode

有两种方式可以设置Activity的启动模式

<!--standard singleInstance singleTask singleTop-->
<activity
    android:launchMode="standard" 
    android:name=".MainActivity">
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />

        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity>

可选值有standard singleInstance singleTask singleTop四种,分别对应上述四种启动模式。

Intent intent = new Intent();
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(new Intent(MainActivity.this,SecondActivity.class));        

其中有两个常用的标记位:Intent.FLAG_ACTIVITY_SINGLE_TOP对应singleTop模式和Intent.FLAG_ACTIVITY_NEW_TASK对应singleTask模式。

对于singleTopsingleTask这两种启动模式的具体区别与使用场景推荐阅读:SingleTop与SingleTask在实际应用中的微妙之处
简单来说就是singleTopsingleTask都无法用来启动自己。singleTop多用于为防止快速多次点击而多次启动Activity;由于singleTask模式的Activity重新启动时会将覆盖在其上层的Activity都销毁掉,所以多用于登录页或App的主页。例如QQ退出登录后进入登录页面,当你按返回键后将会返回手机菜单页面,而不是你点击退出按钮的那个页面。

四、IntentFilter匹配规则

IntentFilter直译过来就是意图过滤器,我们可以通过它的匹配规则去打开我们想要打开的一类Activity,例如我们想要打开手机浏览器,但是我们不知道用户安装了哪些浏览器或者习惯于使用哪个浏览器,那么我们就可以通过IntentFilter来启动,让用户自己选择使用哪个浏览器。
IntentFilter可以在AndroidManifest.xml中注册Activity时通过<intent-filter>标签来设置intentFilter,它有3个标签属性action,categorydata

<activity
  android:launchMode="standard"
  android:name=".MainActivity">
  <intent-filter>
       <action android:name="android.intent.action.MAIN" />

       <category android:name="android.intent.category.LAUNCHER" />
   </intent-filter>
</activity>

在说intentFilter的匹配规则前,有必要得先讲一下Activity的调用模式,注意是调用模式而不是启动模式。Activity的调用模式有两种:显式调用隐式调用

Intent intent = new Intent(FirstActivity.this,SecondActivity.class); 
startActivity(intent);

其实严格来讲,这个也不算是显式调用,因为在显式调用的意义中需要明确之处被启动的对象的组件信息,包括包名和类名,这里并没有之处包名:

Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
ComponentName cn = new ComponentName("com.mg.axe.testappa","com.mg.axe.testappa.MainActivity");
intent.setComponent(cn);
startActivity(intent);  
Intent intent = new Intent(); 
intent.setAction("android.intent.action.View"); 
startActivity(intent);

当我们进行Activity的隐式调用时,IntentFilter就可以排上用场了,那么下面将详细介绍其匹配规则:

<!--SecondActivity的intent-filter-->
<intent-filter>
 <category android:name = "android.intent.category.DEFAULT" />
 <action android:name="com.axe.mg.what" />
</intent-filter>

<!--ThirdActivity的intent-filter-->
<intent-filter>
 <category android:name = "android.intent.category.DEFAULT" />
 <action android:name="com.axe.mg.what" />
 <action android:name="com.axe.mg.how"/>
</intent-filter>

<!--FourthActivity的intent-filter-->
<intent-filter>
 <category android:name = "android.intent.category.DEFAULT" />
 <action android:name="com.axe.mg.why" />
 <action android:name="com.axe.mg.how"/>
</intent-filter>
Intent intent = new Intent();
intent.setAction("com.axe.mg.what");
startActivity(intent);

这种启动方式既可以启动SecondActivity,也可以启动ThirdActivity,但是无法启动FourthActivity。且必须至少含有一个<category android:name = "android.intent.category.DEFAULT" />标签,否则系统会抛出ActivityNotFoundException的异常

<!--SecondActivity的intent-filter-->
<intent-filter>
 <action android:name="com.axe.mg.what" />
 <category android:name="com.yu.hu.category1"/>
 <category android:name="com.yu.hu.category2"/>
 <category android:name = "android.intent.category.DEFAULT" />
</intent-filter>

<!--ThirdActivity的intent-filter-->
<intent-filter>
 <action android:name="com.axe.mg.what" />
 <category android:name = "android.intent.category.DEFAULT" />
 <category android:name="com.yu.hu.category1"/>
 <category android:name="com.yu.hu.category2"/>
 <category android:name="com.yu.hu.category3"/>
</intent-filter>

<!--FourthActivity的intent-filter-->
<intent-filter>
 <action android:name="com.axe.mg.what" />
 <category android:name = "android.intent.category.DEFAULT" />
 <category android:name="com.yu.hu.category2"/>
</intent-filter>
Intent intent = new Intent();
intent.addCategory("com.yu.hu.category1");
intent.addCategory("com.yu.hu.category2");
intent.setAction("com.yu.hu.what");
startActivity(intent);

此时依然只能匹配到前两个Activity,因为FourthActivity没有category1


另外这里还有两点要注意
  1. 因为强制要求一个Activity需要一个<category android:name="android.intent.category.DEFAULT"/>,所以我们不用将这个categoty添加到intent中去匹配。
  2. 如果单独只addCategory是没有用的,必须setAction之后才行。
  • scheme:URI的模式,如http。如果URI中没有指定scheme,那么整个URI无效。默认为contentfile
  • host:URI的host(域名、网址),如www.baidu.com。如果指定了schemeportpath等其他参数,但是host未指定,那么整个URI无效;如果只指定了scheme,没有指定host和其他参数,URI是有效的。
  • port:URI端口,当URI指定了schemehost 参数时port参数才有意义。
  • path:用来匹配完整的路径,如:http://example.com/blog/abc.html,这里将 path 设置为 /blog/abc.html 才能够进行匹配;
  • pathPrefix:用来匹配路径的开头部分,拿上面的 URI 来说,这里将pathPrefix设置为 /blog 就能进行匹配了;
  • pathPattern:用表达式来匹配整个路径。

总的来说有点像是正则表达式,用于匹配指定字段内容。
示例:假如我想要匹配https://www.baidu.com:8080/imgs/*,那么data应该这么写:

<intent-filter>
   <action android:name="xx" />
   <category android:name="android.intent.category.DEFAULT" />
   <data
     android:host="www.baidu.com"
     android:pathPrefix="/imgs"
     android:port="8080"
     android:scheme="https" />
 </intent-filter>

java代码:

  Intent intent = new Intent();
  intent.setData(Uri.parse("https://www.baidu.com:8080/imgs/img1.png"));
  startActivity(intent);

推荐阅读:intent-filter的action,category,data匹配规则

上一篇下一篇

猜你喜欢

热点阅读