最近在iOS上又出现了一个热更新的工具,rollout,小小的体验了一番。
简介
它在编译期间执行一个upload_dsym的脚本,将收集到的文件符号dsym上传到其服务器,这样我们就可以在它提供的后台操作dashboard中,选择对应的app,选择已存在的方法,进行编辑,更改返回值,替换参数,添加逻辑等等。还可以添加新的方法,但是类必须存在,还能为缺失的方法提供实现。
主要技术也是JavaScriptCore和Runtime。在dashboard中添加hot patch之后,客户端重新启动,会拉取新的配置。但在试验中,不是很及时,要启动几次才能切到新的配置,可能跟服务器在国外有关,还有可能就是拉取到配置之后,要patch的动作已经完成,这种在下次启动就可以了。当在工程中新添加类/方法时,在启动后也会将符号上传。但是不能立马在dashboard中看到对应的类。问了rollout的工作人员,说是会有几分钟的延迟,可以过段时间刷新页面看看。
安装
可以通过Cocoapods或者下载他们提供的工具来安装SDK。
我是通过Cocoapods。
另外,我们可以添加options,输出调试信息。
RolloutOptions *option = [[RolloutOptions alloc] init];
option.verbose = RolloutOptionsVerboseLevelDebug;
[Rollout setupWithKey:@"5714eaa036053a902868e088"
#ifdef DEBUG
developmentDevice:YES
#endif
options:option];
下午操作的时候按照它的提示,在前5步完成之后,compile的时候出现了错误。Build Phase-->Rollout.io dsym upload中路径不存在的问题,后来跟rollout的人反馈了这个问题,回复说是个bug,会在下次release中fix。但是现在晚上再重试已经修复了。真有效率啊。
在运行起来之后,Waiting For App会变成Done。可以进行下一步添加hot patch了。
添加hot patch
1、你可以搜索想要的方法名。
为了方便,我们选择[ViewController viewDidLoad]方法。
- toggleJS是让我们自己添加js代码,写逻辑。
- Before the function有2种操作,一种是弹个alert,可以自己设置title,msg等。另一种是进行事件统计。
- ConditionalPatch是添加patch触发的条件,可设置系统版本,手机型号。
- HotPatch Type是是否允许使用patch。
下面我们就简单的选择Before the function添加个alert。
然后,重新运行。有时候并不能马上看到效果,因为它是异步去拉取新的配置,配置拉取回来之后,在下次启动时生效。部分配置如下。
{
"__v" = 0;
"_id" = 5714f968cfbc4acb63bdc419;
"app_version" = 5714eeb692e27ab318a043ba;
bucket = sandbox;
"creation_date" = "2016-04-18T15:12:40.656Z";
data = (
{
class = Test;
configuration = "Ui5OU0NsYXNzRnJvbVN0cmluZygnVUlBcHBsaWNhdGlvbicpLmNhbGwoJ3NoYXJlZEFwcGxpY2F0aW9uJykuY2FsbCgnb3BlblVSTDonLCBSLk5TQ2xhc3NGcm9tU3RyaW5nKCdOU1VSTCcpLmNhbGwoJ1VSTFdpdGhTdHJpbmc6JywgJ2h0dHA6Ly93d3cuYmFpZHUuY29tJykpOwo=";
configurationType = javascript;
methodType = instance;
selector = test;
signature = "()->Void";
swizzlingType = createImplementation;
},
{
class = ViewController;
configuration = Ui5OU0NsYXNzRnJvbVN0cmluZygnTXlPYmplY3QnKS5jYWxsKCduZXcnKS5jYWxsKCdzYXknKTsK;
configurationType = javascript;
methodType = instance;
selector = viewDidLoad;
signature = "()->Void";
swizzlingType = replaceImplementation;
}
);
}
运行结果如下:
2、Polyfille method
可以理解为方法替代。只能给已存在的类添加方法。可以添加新的方法,或者是给已声明未实现的方法添加实现。
下面的例子中因为[MyObject test]未实现,在MyViewController里面ViewDidload方法调用了[MyObject test]方法,正常情况下是跳转到MyViewController页面就会崩溃的。等下用hot patch来解决。经测试,如果实现[MyObject test],就会走原有逻辑,patch就不起作用了。如果要修改已存在方法,可参照第三条。
@interface MyObject : NSObject
- (void)say;
// 未实现,之后会用hot patch实现
- (void)test;
- (void)calculate:(int)a;
@end
@implementation MyObject
- (void)say {
NSLog(@"I am MyObject");
}
- (void)calculate:(int)a {
NSLog(@"calculate:%d", a);
}
@end
我们添加了[MyObject test]方法。
然后添加实现。这里我们打开百度页面。js的写法也比较简单。
R.NSClassFromString('UIApplication').call('sharedApplication').call('openURL:', R.NSClassFromString('NSURL').call('URLWithString:', 'http://www.baidu.com'));
调用方法就用call,传参数,直接跟在后面。
运行,跳转到MyViewController,会跳转到web。
3、修改已存在的方法
还是拿[ViewController viewDidLoad]来说,在里面调用方法[MyObject say]。选择Toggler JS。[MyObject say]是在工程中的方法。
var obj = R.NSClassFromString('MyObject').call('new')
obj.call('say')
self.originalImplementation();
再次运行。
4、修改参数值
将calculate参数为10,修改成20。
在ViewController里
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
MyObject *obj = [[MyObject alloc] init];
[obj calculate:10];
}
运行几次,发现的确是输出了20。