草庐IT

#yyds干货盘点# 关于Activity,一个初级开发应该知道的一切

Petterpx 2023-03-28 原文

Activity 作为一个老生常谈的话题,它是我们刚接触Android开发就遇到,虽然已有一段时间开发经验,但谈起完全搞懂Activity相关,不敢妄言,故结合个人理解及书籍参考,简单总结一下 Activity基础相关知识,其中也留出了一些有意思的问题。

大体的脑图如下(启动流程涉及的过多,暂时放在后期):

Activity生命周期

基础的生命周期方法这里就不做解释了,大家刚开始接触时就是这些方法了。

一些有意思的问题

  • onStart 和 onResume,onPuse 和 onStop有什么实质的不同吗?

实际使用来说,他们看起来的确差不多,但是 onStart和onStop 是从Activity是否可见这个角度来回调的,而 onResume 和 onPause是从Actvity是否位于前台这个角度来回调的,除了这两点,实际使用中并无其他区别。

  • 假设当前 Activity为A,如果这时用户打开一个新的Activity B,那么B的onResume和A的onPuase那个先执行?

    A的 onPause 先执行。

    在Android的官方文档中,在旧的Activity onPause执行完之后,新的Activity 才能onResume,所以我们应该尽量避免在 onPause 中做太多耗时操作,尽量应该放到onStop中。

异常情况下的生命周期与处理方式

在我们开发中,经常会遇到转屏的问题,而转屏一般也会带来 Activity的重新创建,所以大多数开发者开发的时候,Activity默认是禁止转屏的,但是在一些短视频软件上,转屏就是一件非常常见的事了,那么如何处理相应数据的保存与恢复就是我们必须关注的事了。

Activity异常下的生命周期图

当系统配置发生改变后,Activity会被销毁,其 onPause,onStop,onDestory均会被调用,同时由于Activity是在异常情况下终止的,系统会调用 onSaveInstanceState 来保存当前 Activity 的状态。这个方法的调用时机是在 onStop 之前,它和onPause 没有既定的时序关系,有可能在onPause之前调用,也有可能在 onPause之后调用。但需要注意的是,这个方法只会出现在 Activity 被异常终止的情况下。正常情况下不会回调这个方法。

当Actiivty 被重新创建后,系统会调用 onRestoreInstanceState, 并且吧 Activity 销毁时 onSaveInstanceState 方法保存的 Bundle 对象作为参数同时传递给 onRestoreInstanceSate 和 onCreate 方法。因此我们可以通过 onRestoreInstanceState和 onCreate 方法来判断 Activity是否被重建了,如果被重建了,那么我们就可以去除之前保存的数据并恢复,从时序上来说,onRestoreInstanceState 的调用时机在 onStart之后。

onSaveInstanceState与ViewModel

在上面我们知道,当Activity因为异常情况发生重建时,系统会主动调用 onSaveInstanceState 方法来进行保存,但需要注意的是 onSaveInstanceState 并不适合于保存大量数据,Google的推荐是用其来保存相应的id及key,而相应的大量数据推荐使用ViewModel进行保存。

既然使用ViewModel进行保存了,那 onSaveInstanceState 的意义还有什么?

  • onSaveInstanceState 是为了保存应用进程在后台时候由于内存限制而被终止,或者配置更改时的回调,其 默认实现是保存了关于 activity的视图层次状态的临时信息,比如EditText中输入的文本,Rv,Lv的滑动位置等,其支持的类型只是Bundle,所以并不适合存储大量数据,适合于少量的临时数据。

  • ViewModel 可以代理复杂数据的加载,也可以作为临时的存储位置,但是不能在手动 finish 的进程中存留,它的意义更多的是实现 当系统状态更改时,实现数据的保留,而不是ui状态的保留。


Activity的启动模式

为什么Activity需要启动模式呢?
在默认情况下,当我们多次启动同一个Activity 时,系统会创建多个实例并把他们按照 后进先出的原则(栈结构) 一一放进任务栈中。任务栈是一种 后进先出 的栈结构。每次返回时都会销毁一个 Activity 。所以为了更好的管理Activity 和对Activity进行利用,Android提供了 启动模式来解决这个问题,分别为standard,SingleTop,singleTask,SingleInstance

standard

标准模式,也是默认模式。即每次启动一个Activity 都会重新创建一个新的实例,不论这个实例是否存在。相应的生命周期也遵从标准的生命周期过程。

singleTop

栈顶复用模式。简单理解为,如果新的Activity采用这个模式启动,如果此Activity已经处于当前任务栈栈顶,那么此Activity不会被重复创建,当调用 startActivity跳转时,会回调它的 onNewIntent() 方法。通过此方法我们可以取出当前请求的信息。需要注意的是如果使用 startActivityForResult 跳转,将忽略启动模式。

singleTask

栈內复用模式。这是一种单实例模式,在这种模式下,只要Activity在一个栈中存在,那么多次启动此Activity都不会创建实例,和 singleTop 一样,系统也会回调 onNewIntent. 简单理解为,当启动一个 启动模式为 singleTask的Activity时,系统会再栈里寻找是否存在此Activity,如果找到,将此Activity顶部的所有Activity全部出栈,并把其调到栈顶并调用它的 onNewIntent 方法。如果不存在此Activity,则创建此Activity并压入栈顶。

singleInstance

单实例模式,又称加强的 singTask 模式。它除了具有singleTask的所有特性外,还具有额外的特性,那就是 具有此启动模式的Activity 只能单独位于一个 任务栈中。

Activtiy的Flags

Activity的Flags有很多,这些标记位在我们实际开发中帮助很大,其中有些标记位可以设定 Activity的启动模式,比如 使用 Application 启动Activity时 添加的 FLAG_ACTIVITY_NEW_TASK 等。

FLAG_ACTIVITY_NEW_TASK

为Activity指定 singleTask 的启动模式. 问题为什么Application调用startActivity需要附带此tag?

FLAG_ACTIVITY_SINGLE_TOP

为Activity指定 singleTop 的启动模式

FLAG_ACTIVITY_CLEAR_TOP

销毁目标 Actiivty 和它之上的所有Activity并重新创建。如果搭配 FLAG_ACTIVITY_SINGLE_TOP 使用,会销毁它之上的所有Activity,如果目标实例存在,则不会销毁,会调用其的 onNewIntent 方法。

FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS

具有这个标记的 Activity不会出现在历史Activity列表,等同于 xml中指定Activity的属性 android:excludeFromRecents="true"。

注意,这个参数如果放在首个Activity,那么接下来这个Activity栈的所有Activity都会受到影响。


<br/>

IntentFilter的匹配规则

启动一个Activity分为两种,显式和隐式,日常开发中我们用的最多的就是显式,而隐式常常用作,h5打开一个app,或者调用系统某个设置页,打开相机等等。隐式调用相比显式调用来说,稍微复杂一点,它需要Intent能够匹配目标组件IntentFilter 中所设置的过滤信息,如果不匹配将无法启动目标Activity。 IntentFilter 中的过滤信息 有 action,category,data

action的匹配规则

action是一个字符串,系统预定义了一些action,同时我们也可以定义我们自己的action。action的匹配规则是Intent中的action必须能够和过滤规则中的action匹配。而过滤规则我们可以定义多个,只要Intent中的action能与任意一个过滤规则匹配就是匹配成功。如果你的Intent中没有定义action,则匹配失败。

<action android:name="android.intent.action.VIEW" /> 常用的action如下:

  • ACTION_MAIN android.intent.action.MAIN 应用程序入口
  • ACTION_VIEW android.intent.action.VIEW 显示数据给用户
  • ACTION_ATTACH_DATA android.intent.action.ATTACH_DATA 指明附加信息给其他地方的一些数据
  • ACTION_EDIT android.intent.action.EDIT 显示可编辑的数据
  • ACTION_PICK android.intent.action.PICK 选择数据
  • ACTION_CHOOSER android.intent.action.CHOOSER 显示一个Activity选择器
  • ACTION_DIAL android.intent.action.GET_CONTENT 显示打电话面板
  • ACITON_CALL android.intent.action.DIAL 直接打电话
  • ACTION_SEND android.intent.action.SEND 直接发短信
  • ACTION_SENDTO android.intent.action.SENDTO 选择发短信
  • ACTION_ANSWER android.intent.action.ANSWER 应答电话

category匹配规则

category是一个字符串,系统也为我们预制了一席,对于在 已经定义的匹配规则,在Intent 中存在的categoty必须全部符合已经定义了的规则,当然也可以不填,如果Intent中没有包含,系统会为我们默认带上 android.intent.category.DEFAULT,这一点和action 有一些区别.

<category android:name="android.intent.category.DEFAULT" />

data 匹配规则

data的匹配规则和 action 类似,如果定义了相应的匹配规则,那么Intent中必须包含相应的data.

<data android:host="*" android:mimeType="string" android:path="/test" android:pathPattern="*" android:pathPrefix="/petterp" android:port="1080" android:scheme="http" /> data由两部分组成,mimeType和URI,mimeType指媒体类型,比如image/jpeg. Audio/mpeg4-genric和 video/*等,可以表示图书,文本,视频等不同的媒体形式。而URI中包含的数据就相对多一点。

Uri的结构如下:

<scheme>://<host>:<port>/[<path>|<pathPrefix>|<pathPattern>]
  • Scheme: URI 的模式,比如 http,file,content等,如果 URI中没有指定 scheme.那么整个 URI的其他参数无效,这也意味着URI是无效的。
  • Host: URI的主机名,比如 www.baidu.com.如果host未指定,那么整个Uri中的其他参数无效,这也意味着URI是无效的。
  • Port:URI中的端口号,比如 80,仅当URI 中指定了 scheme 和 host参数的时候,port参数才是有意义的。Path,pathPattern和 pathPrefix这三个参数表示路径信息。
    • 其中path表示完整的路径信息,
    • pathPattern 也表示完整的路径信息,但是它里面可以包含通配符 ***** ,***** 表示0个或多个任意字符,需要注意的是,由于正则表达式的规范,如果想要表示真实的字符串,那么 ***** 要写成 \\* ,\ 要写成 \\\
    • pathPrefix表示路径的前缀信息。

data的匹配规则

data的匹配规则和action 类似,它也要求 Intent 中必须含有 data数据,并且 data数据能够完全匹配过滤规则中的某一个 data,这里的完全匹配是指 过滤规则中出现的 data部分也出现在了 Intent 中的 data中。

如下所示匹配规则:

<intent-filter> <action android:name="petterp.test1" /> <data android:mimeType="image/*" /> </intent-filter> 这种匹配规则指定了媒体类型为所有类型的图片,那么Intent中的 mineType 属性必须为 image/* 才能匹配,这种情况下虽然过滤规则没有指定URI,但是却有默认值。URI 的默认值为 contentfile 。也就是说,虽然没有指定 URI,但是Intent中的 URI 部分的 schema 必须为 content 或者 file才能匹配。

所以如果我们要调用的话,可以写出如下的代码:

val intent=Intent("petterp.test1") intent.setDataAndType("file://abc","iamge/*") 接着,我们再看一个案例,相比上面的,复杂一点

<intent-filter> <action android:name="petterp.test2" /> <category android:name="android.intent.category.DEFAULT" /> <data android:host="www.baidu.com" android:mimeType="image/*" android:path="/test" android:port="8080" android:scheme="https" android:pathPattern=".*"/> </intent-filter> 对于这个案例,我们在调用的时候,就必须采用以下格式:

val intent = Intent("petterp.test2") intent.setDataAndType(Uri.parse("https://www.baidu.com:8080/test/路径随便"),"image/*") 如果这里看不懂的话,多看一遍data匹配规则就行。

注意:category必须加,就算Intent(intent中默认会带一个)中不加,xml中也必须带,否则会存在找不到相应的Activity,原因是因为不含有 DEFAULT 这个 category的Action是无法接收Intent的。

非常高兴你能看到这里,虽然这只是Activity的一些基础,但都是我们每个开发者最基本应该掌握的。勉励前行

参阅资料

  • 《Android开发艺术探索》
  • Google开发者说

有关#yyds干货盘点# 关于Activity,一个初级开发应该知道的一切的更多相关文章

  1. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  2. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  3. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  4. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

    为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

  5. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  6. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

  7. ruby - 为什么 SecureRandom.uuid 创建一个唯一的字符串? - 2

    关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?

  8. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  9. ruby-on-rails - Rails - 从另一个模型中创建一个模型的实例 - 2

    我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案

  10. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

随机推荐