diff --git a/blogs/Android/Activity.md b/blogs/Android/Activity.md
index 30d0342..ed0644d 100644
--- a/blogs/Android/Activity.md
+++ b/blogs/Android/Activity.md
@@ -7,12 +7,372 @@ Activity
1. 思维导图
2. 概述
3. 生命周期
+ - 正常情况下的生命周期(注意透明 Activity 和 DialogActivity)
+ - 异常情况下的生命周期
4. 启动模式
-5. 重要方法
-6. 转场动画
-7. 参考
+ - LaunchMode
+ - standard
+ - singleTop
+ - singleTask
+ - singleInstance
+ - Intent.Flags
+ - FLAG_ACTIVITY_NEW_TASK
+ - FLAG_ACTIVITY_SINGLE_TOP
+ - FLAG_ACTIVITY_CLEAR_TOP
+ - FLAG_ACTIVITY_EXCLUDE_FORM_RECENTS
+5. IntentFilter 匹配规则
+ - action
+ - category
+ - data
+ - URI
+ - mimeType
+6. 重要方法
+ - onSaveInstanceState、onRestoreInstanceState
+ - startActivityForResult
+ - onNewIntent
+ - onConfigurationChanged
+ - onTirmMemory
+7. 转场动画
+ - overridePendingTransition
+ - 设置 Application Stype
+ - ActivityOptions
+8. 启动流程
+ 1. Launcher 请求 AMS
+ 2. AMS 到 ApplicationThread 的调用过程
+ 3. ActivityThread 启动 Activity
+9. 参考
#### 思维导图
+![](https://i.loli.net/2019/01/31/5c52b6d319de9.png)
+
#### 概述
+Activity 即用户界面。
+
+#### 生命周期
+
+说到生命周期,最经典的一张图:
+
+![](https://i.loli.net/2019/01/30/5c518e91a1905.png)
+
+不过还是说一下实例会更好理解一下:
+
+| 情况 | 回调 |
+| ----------------------- | ------------------------------------------------------------ |
+| 第一次启动 | onCreate 、onStart、onResume |
+| 从 A 跳转到不透明的 B | A_onPause、B_onCreate、B_onStart、B_onResume、A_onStop |
+| 从 A 跳转到透明的 B | A_onPause、B_onCreate、B_onStart、B_onResume |
+| 从不透明的 B 再次回到 A | B_onPause、A_onRestart、A_onStart、A_onResume、B_onStop、B_onDestory |
+| 从透明的 B 再次回到 A | B_onPause、A_onResume、B_onStop、B_onDestory |
+| 用户按 home 键 | onPause、onStop |
+| 按 home 键回到应用 | onRestart、onStart |
+| 用户按 back 键会退 | onPause、onStop、onDestory |
+
+透明 Activity 和 DialogActivity 类似,在跳到 DialogActivity 也不会回掉前一个 Activity 的 onStop 方法。
+
+#### 启动模式
+
+##### LaunchMode
+
+| 启动模式 | 说明 |
+| -------------- | ------------------------------------------------------------ |
+| standard | 标准模式,也是系统的默认模式,每次启动一个 Activity 都会重新创建一个实例 |
+| singleTop | 栈顶复用,如果新 Activity 已经位于任务栈的栈顶,那么此 Activity 不会被重新创建,同时它的 onNewIntent 方法会被回调 |
+| singleTask | 栈内复用,只要 Activity 在一个栈中存在,那么多次启动此 Activity 都不会重新创建实例,和 singleTop 一样,系统也会回调其 onNewIntent |
+| singleInstance | 单例模式,启动的 Activity 会创建一个新的任务栈并压入栈中,由于栈内复用的特性,后续的请求均不会创建新的 Activity,除非这个任务栈被系统销毁了 |
+
+在 singleTask 启动模式中,多次提到某个 Activity 所需的任务栈,什么是 Activity 所需要的任务栈呢?这就要从一个参数说起:taskAffinity,任务相关性。这个参数标识了一个 Activity 所需要的任务栈的名字,默认情况下,所有 Activity 所需的任务栈的名字为应用的包名。当然,我们可以为每个 Activity 都单独指定 taskAffinity 属性,这个属性值必须不能和包名相同,否则相当于没有设置。taskAffinity 属性主要和 singleTask 启动模式和 allowTaskReparentiong 属性配对使用,在其他情况下没有意义。
+
+taskAffinity 与 singleTask 配对使用:
+
+如果启动了设置了这两个属性的 Activity,这个 Activity 就会在 taskAffinity 设置的任务栈中。
+
+taskAffinity 与 allowTaskReparenting 配对使用:
+
+当一个应用 A 启动了应用 B 的某个 Activity 后,如果这个 Activity 的 allowTaskReparenting 属性为 true 的话,那么当应用 B 被启动后,此 Activity 会直接从应用 A 的任务栈转移到应用 B 的任务栈中。这个属性主要作用就是将这个 Activity 转移到它所属的任务栈中,例如一个短信应用收到一个带有网络链接的短信,点击链接会跳到浏览器,这时候如果 allowTaskReparenting 设置为 true 的话,打开浏览器应用就会直接显示刚才打开的网页页面,而打开短信应用后这个浏览器界面就会消失。
+
+启动模式了解之后,那是如何指定启动模式的方式呢?
+
+有两种:一种是在 AndroidMenifet 文件设置 launchMode 属性,一种是给 Intent 设置 Flag。
+
+如果两者都存在,后者优先级更高。
+
+##### Activity 中的 Flags
+
+| 标记位 | 说明 |
+| ---------------------------------- | ------------------------------------------------------------ |
+| FLAG_ACTIVITY_NEW_TASK | 为 Activity 指定 singleTask 启动模式,其效果和在 xml 中指定启动模式相同 |
+| FLAG_ACTIVITY_SINGLE_TOP | 为 Activity 指定 singleTop 启动模式,其效果和在 xml 中指定启动模式相同 |
+| FLAG_ACTIVITY_CLEAR_TOP | 具有此标记位的 Activity,当启动它时,同一个任务栈中所有位于它上面的 Activity 都要出栈。这个模式一般需要和 FLAG_ACTIVITY_NEW_TASK 配合使用,在这种情况下,被启动的 Activity 实例如果已经存在,那么系统就会调用它的 onNewIntent。如果被启动的 Activity 采用 standard 模式启动,那么它连同它之上的 Activity 都要出栈,系统会创建新的 Activity 实例并放入栈顶 |
+| FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS | 具有这个标记的 Activity 不会出现在历史的 Activity 的列表中,当某些情况下我们不希望用户通过历史列表回到我们的 Activity 的时候这个标记比较有用。它等同于在 xml 中指定 Activity 的属性 android:excludeFormRecents="true" |
+
+#### IntentFilter 的匹配规则
+
+启动 Activity 分为两种,显式调用和隐式调用。显式调用就不多说了,隐式调用需要 Intent 能够匹配目标组件的 IntentFilter 中所设置的过滤信息,如果不匹配将无法启动目标 Activity。IntentFilter 的过滤信息有 action、category、data。
+
+IntentFilter 需要注意的地方有以下:
+
+1. 一个 Activity 中可以有多个 intent-filter
+
+2. 一个 intent-filter 同时可以有多个 action、category、data
+
+3. 一个 Intent 只要能匹配任何一组 intent-filter 即可启动对应 Activity
+
+4. 新建的 Activity 必须加上以下这句,代表能够接收隐式调用
+
+ ```java
+
+ ```
+
+##### action 的匹配规则
+
+action 的匹配规则就是只要满足其中一个 action 就可以启动成功。
+
+```xml
+
+
+
+
+
+
+
+```
+
+```java
+Intent intent = new Intent();
+intent.setAction("demo");
+startActivity(intent);
+```
+
+注意,action 是区分大小写的。
+
+##### category 匹配规则
+
+category 在代码设置如下:
+
+```java
+intent.addCategory("")
+```
+
+这句可以添加也可以不添加,因为代码默认会为我们匹配 “android:intent.category.DEFAULT”。
+
+##### data 匹配规则
+
+data 主要是由 URI 和 mimeType 组成的。URI 的结构如下:
+
+```
+://:[||]
+```
+
+语法如下:
+
+```xml
+
+```
+
+实例:
+
+```java
+
+
+
+
+
+
+
+```
+
+```java
+ Intent intent = new Intent();
+ intent.setData(Uri.parse("demo://omooo:80/data?age=18&sex=girl"));
+ if (getPackageManager().resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY) != null) {
+ startActivity(intent);
+ } else {
+ Toast.makeText(this, "Activity Not Found", Toast.LENGTH_SHORT).show();
+ }
+```
+
+这里,注意我用了 PackageManager#resolverActivity() 来判断是否存在该隐式 Activity,这样就避免找不到 Activity 而导致应用 Crash。同时注意,第二个参数必须传 PackageManager.MATH_DEFAULT_ONLY,因为只有 category 为 DEFAULT 的 Activity 才能接收隐式启动。
+
+```java
+ if (getIntent().getData() != null) {
+ Uri uri = getIntent().getData();
+ String scheme = uri.getScheme();
+ Log.i(TAG, "scheme: " + scheme);
+ String host = uri.getHost();
+ Log.i(TAG, "host: " + host);
+ int port = uri.getPort();
+ Log.i(TAG, "port: " + port);
+ String path = uri.getPath();
+ Log.i(TAG, "path: " + path);
+ String query = uri.getQuery();
+ Log.i(TAG, "query: " + query);
+ Set params = uri.getQueryParameterNames();
+ for (String item : params) {
+ Log.i(TAG, "item: " + item + " value: " + uri.getQueryParameter(item));
+ }
+ }
+//输出:
+Demo_SecondActivity: scheme: demo
+Demo_SecondActivity: host: omooo
+Demo_SecondActivity: port: 80
+Demo_SecondActivity: path: /data
+Demo_SecondActivity: query: age=18&sex=girl
+Demo_SecondActivity: item: age value: 18
+Demo_SecondActivity: item: sex value: girl
+```
+
+说完 URI,剩下的 mimeType 表示要传递的数据类型,通常是 text/plain 或 image/jpeg 等等。
+
+```java
+intent.setType("text/plain");
+```
+
+不过需要注意的是,如果同时设置了 URI 和 mimeType 的话就必须使用以下代码才能跳转:
+
+```
+intent.setDataAndType(Uri.parse("demo://omooo"),"text/plain")
+```
+
+因为如果使用 setData() 或者 setType() 的话,分别会将相应的 type 和 data 置为 null。
+
+#### 重要方法
+
+1. onSaveInstanceState()、onRestoreInstanceState()
+
+ 这两个方法只有在应用遇到意外情况下才会触发,比如横竖屏切换,可以用于保存一些临时性数据。
+
+ 在横竖屏切换的时候,Activity 会重建,在 onStop 之前,和 onPause 没有既定的时序关系,会调用 onSaveInstanceState(Bundle bundle) 方法,其中有一个 Bundle 对象可以用来存储数据,该对象便是 onCreate 中的 Bundle 对象 savedInstancesState。在 onCreate 取得时候要注意判空,而在 onRestoreInstanceState 里面则不需要判空,onRestoreInstanceState 方法调用的时机是在 onStart 之后。
+
+ 同时,我们知道,在 onSaveInstanceState 和 onRestoreInstanceState 方法中,系统自动为我们做了一定的恢复工作。当 Activity 在异常情况下需要重新创建,系统会默认为我们保持当前 Activity 的视图结构,并且在 Activity 重启后为我们恢复这些数据,比如文本框中用户输入的数据、ListView 滚动的位置等等。具体针对某一个特定的 View 系统能为什么恢复那些数据,可以查看 View 的源码。和 Activity 一样,每个 View 都有 onSaveInstanceState 和 onRestoreInstanceState 这两个方法。
+
+ 关于保存和恢复 View 层次结构,系统的工作流程是这样的:首先 Activity 被意外终止时,Activity 会调用 onSaveInstanceState 去保存数据,然后 Activity 会委托 Window 去保存数据,接着 Window 在委托它上面的顶级容器再去一一通知它的子元素来保存数据,这样整个数据保存过程就完成了。
+
+ ```java
+ @Override
+ public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState) {
+ super.onSaveInstanceState(outState, outPersistentState);
+ }
+
+ @Override
+ protected void onRestoreInstanceState(Bundle savedInstanceState) {
+ super.onRestoreInstanceState(savedInstanceState);
+ }
+ ```
+
+2. startActivityForResult
+
+ FirstActivity:
+
+ ```java
+ public void skip(View view) {
+ Intent intent = new Intent(this, SecondActivity.class);
+ startActivityForResult(intent, 0x01);
+ }
+
+ @Override
+ protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
+ super.onActivityResult(requestCode, resultCode, data);
+ if (requestCode == 0x01 && resultCode == 0x02
+ && data != null) {
+ Toast.makeText(this, data.getStringExtra("name"), Toast.LENGTH_SHORT).show();
+ }
+ }
+ ```
+
+ SecondActivity:
+
+ ```java
+ public void goBack(View view) {
+ Intent intent = new Intent();
+ intent.putExtra("name", "Omooo");
+ setResult(0x02, intent);
+ finish();
+ }
+ ```
+
+ 这里需要注意的是,requestCode 一定要为正数,看源码便知。其实 startActivity 内部也是调用 startActivityForResult,然后在 startActivityForResult 的第二个参数 requestCode 传 -1。
+
+3. onNewIntent
+
+ ```java
+ @Override
+ protected void onNewIntent(Intent intent) {
+ super.onNewIntent(intent);
+ }
+ ```
+
+ 在启动模式那块说过,当启动模式是 singleTask 和 singleTop 的时候,系统不需要重新创建 Activity 的时候会回调 onNewIntent 方法。
+
+4. onConfigurationChanged
+
+ 前面说过,当 Activity 横竖屏切换的时候会导致 Activity 销毁并重建,哪有什么方法能避免呢?其实可以在 AndroidManifest 里面指定 android:configChanges="orientation/screenSize" 来避免重建,这时就会调用 onConfigurationChanged 方法。
+
+ 如果按上面的配置,当字体发生变化时,也会销毁重建,但是不会回调 onConfigurationChanged 方法,所以说想要监听的变化必须要包含之内。
+
+ ```java
+ @Override
+ public void onConfigurationChanged(Configuration newConfig) {
+ super.onConfigurationChanged(newConfig);
+ if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) {
+ //竖屏
+ } else {
+ //横屏
+ }
+ }
+ ```
+
+5. onTrimMemory
+
+ ```java
+ @Override
+ public void onTrimMemory(int level) {
+ super.onTrimMemory(level);
+ if (level == TRIM_MEMORY_UI_HIDDEN) {
+
+ }
+ }
+ ```
+
+ 当内存紧张时会回调,它在 onStop 回调之前。指导应用程序在不同的情况下进行自身的内存释放,以避免被系统直接杀掉,提高应用程序的用户体验。它和 onLowMemory 相比,它有一个 level 评级,onLowMemory 能兼容更低的版本。
+
+#### 转场动画
+
+1. overridePendingTransition(int enterAnim,int exitAnim)
+
+ 需要注意,在 startActivity 和 finish 之后调用。
+
+2. 设置 Application style
+
+3. ActivityOptions
+
+#### 启动流程
+
+Activity 的启动流程可以分为三个部分:
+
+1. Launcher 请求 AMS 过程
+2. AMS 到 ApplicationThread 的调用过程
+3. ActivityThread 启动 Activity
+
+#### 参考
+
+《Android 开发艺术探索》
+
+[Android面试官装逼失败之:Activity的启动模式](https://juejin.im/post/59b0f25551882538cb1ecae1)
+
+[Github项目解析(九)-->实现Activity跳转动画的五种方式](https://blog.csdn.net/qq_23547831/article/details/51821159)
+
+[https://github.com/HellForGate/TransitionDemo](https://github.com/HellForGate/TransitionDemo)
+
+
+
diff --git a/images/Android/Activity.png b/images/Android/Activity.png
new file mode 100644
index 0000000..8dc42a4
Binary files /dev/null and b/images/Android/Activity.png differ
diff --git a/images/Android/activity_lifecycle.png b/images/Android/activity_lifecycle.png
new file mode 100644
index 0000000..ab0921a
Binary files /dev/null and b/images/Android/activity_lifecycle.png differ