iOS 动态更新

App 动态更新

  • 一、控件到 window 的层级关系:git

  • 二、分析控件的详细路径:github

  • 三、动态修改控件:算法

  • 四、工具篇:windows

视图的层级关系

每一个 App , 至少有一个根 Window , 一般状况下咱们只用一个 。window 有一个 rootViewController , 这就是咱们所谓的根视图 , 咱们全部的控制器都是放在 rootViewController 里面的。数组

Dynamic-image1.jpeg

这个是最简单的层级关系服务器

若是在项目里有了这么一个路径 , 咱们能够作什么呢?app

  • 在当项目很复杂 , 能够其它地方能够直接修改这个控件的状态函数

  • 当某个控件命名存在却又没有显示出来 , 能够经过路径来辅助查找工具

  • 由服务器下发一些配置 , 使用 Runtime 去动态的修改已上线的项目布局

下面将介绍如何使用代码来找出这些视图(控件)的路径

分析控件的详细路径

一、找出根 Window :

每个视图、控件 , 他们最终的根都是main函数返回的 application , 经过 [UIApplication sharedApplication] 能够获得 。 applicationwindows 属性是一个数组 , 这里面装的是这个应用的全部 Window , 咱们一般用的是第一个也就是 application.windows[0]

二、遍历视图 :

获得了 window 对象一切都好办了 。 而后拿到 windowrootViewController , 在获取 rootViewController 里面全部的 childViewControllersview 里的 subviews , 一直递归下去就能够获得当前屏幕里全部视图对象了 , 同时能够经过 runtime 把它们的 propertydelegate 都获取出来。

结合 Reveal 或者 Xcode 自带的 Captuer View Hiearachy , 咱们能够推测一下这两个的的实现原理了 :

一、根据应用获得根视图
二、递归获取里面的全部控件
三、按照他们的层级关系一层一层的画出来

动态修改控件

一、把上面获取到的全部控件的详细信息上传到服务器 。
二、根据业务需求由服务器给咱们下发对应的配置列表,以 button 为例 : 配置列表里必需要有 :

1)、button 的全路径 : 如 UIWindow -> UIWindow -> UIView -> UIView -> UILayoutContainerView -> UITabBar -> UIView —> UIButton
2)、button 的惟一标识 : 如 tag 值或本身实现的一套算法生成的惟一标识 , 目的是防止与 button 同一层次的视图搞混 。
3)、 根据路径及惟一标识来匹配 App 里的控件 , 匹配和上面的查找原理是相通的。
4)、 匹配成功表明 button 确实存在 , 根据业务需求作后续操做 。
提示: 匹配策略尽量的多 , 防止意外状况某一两个标识生成失败或者生成相同 。

三、修改 button 的状态。

1)、 如某个按钮点了会 Crash 或暂时不须要被点击 , 可是又要展现出来 , 能够直接修改 buttonenabled 属性 。
2)、 如某业务暂时关闭 , 能够直接修改入口 按钮 frame为0 , 前提是要自动布局已作好 。
3)、 如给购买 按钮 添加监听事件 addTarget: action: forControlEvents:
target 也能够经过上面 遍历视图 获取到 , action 能够由服务器下发 , 也能够一开始就写死 , 等有需求的时候直接传不一样的参数就好了 。

四、 绑定查找控件时 , 这个界面必需要已经初始化完成了才行 , 假如界面还没生成确定是查找不到这个控件的 。 这里给你们提供两种思路 :

一、使用Runtime Method Swizzing , 直接把修改控件的方法与 didMoveToSuperviewdidMoveToWindow 动态绑定 , 等这个控件加载出来以后再去修改 , 查找路径正确的话确定就能找到了 。
二、在具体的类里面 , 等控件的初始化方法调用完后 , 再去执行动态修改 , 如在viewDidLoad 里面初始化控件 , 在 viewWillAppear: 里面动态修改 。
建议使用第一种适用范围更强 。

上架后的 应用 可能会遇到的一些突发情况 , 未测出的Crash、临时改点小需求 , 等等 , 咱们总不能每次由于一点小改动就从新提交一次 App Store , 先不说 App Store 的审核时间 , 频繁的让用户去更新应用 , 用户也会烦的 。使用这篇文章所讲的来实现动态更新是再合适不过了 。

首先上面讲的 动态更新 是彻底脱离出来的一个模块 , 跟业务逻辑没有任何关系 , 只须要部署一次就好了 , 等开发下一个项目也能够直接拿过去使用 。这里的动态更新适用于局部的视图、控件的修改 , 若是你有其它需求能够考虑 JSPatch , 下发脚本也是一个不错的选择 。

工具篇

使用一些UI调试的辅助工具 , 使咱们查看视图在项目中得层次结构更为方便 。
经常使用的UI调试的工具:

  • Captuer View Hiearachy

  • Reveal

Xcode自带的 Captuer View Hiearachy 实现步骤:

  • 一、打开Xcode , 运行项目 , 选择最顶部的 Debug

  • 二、Debug -> View Debugging -> Show View Frames

  • 三、Debug -> View Debugging -> Captuer View Hiearachy

Xcode里面就变成了三维的视图了 , Xcode左侧展现出来的是层级关系的树状图 。

Reveal的功能相对来讲更强大 , 适用于UI调试视图查找 。使用方法请看 Reveal集成指南

相关文章
相关标签/搜索