最近在iOS上又出現(xiàn)了一個熱更新的工具,rollout,小小的體驗了一番。
簡介
它在編譯期間執(zhí)行一個upload_dsym的腳本,將收集到的文件符號dsym上傳到其服務(wù)器,這樣我們就可以在它提供的后臺操作dashboard中,選擇對應(yīng)的app,選擇已存在的方法,進(jìn)行編輯,更改返回值,替換參數(shù),添加邏輯等等。還可以添加新的方法,但是類必須存在,還能為缺失的方法提供實現(xiàn)。
主要技術(shù)也是JavaScriptCore和Runtime。在dashboard中添加hot patch之后,客戶端重新啟動,會拉取新的配置。但在試驗中,不是很及時,要啟動幾次才能切到新的配置,可能跟服務(wù)器在國外有關(guān),還有可能就是拉取到配置之后,要patch的動作已經(jīng)完成,這種在下次啟動就可以了。當(dāng)在工程中新添加類/方法時,在啟動后也會將符號上傳。但是不能立馬在dashboard中看到對應(yīng)的類。問了rollout的工作人員,說是會有幾分鐘的延遲,可以過段時間刷新頁面看看。
安裝
可以通過Cocoapods或者下載他們提供的工具來安裝SDK。
我是通過Cocoapods。
另外,我們可以添加options,輸出調(diào)試信息。
RolloutOptions *option = [[RolloutOptions alloc] init];
option.verbose = RolloutOptionsVerboseLevelDebug;
[Rollout setupWithKey:@"5714eaa036053a902868e088"
#ifdef DEBUG
developmentDevice:YES
#endif
options:option];
下午操作的時候按照它的提示,在前5步完成之后,compile的時候出現(xiàn)了錯誤。Build Phase-->Rollout.io dsym upload中路徑不存在的問題,后來跟rollout的人反饋了這個問題,回復(fù)說是個bug,會在下次release中fix。但是現(xiàn)在晚上再重試已經(jīng)修復(fù)了。真有效率啊。
在運行起來之后,Waiting For App會變成Done。可以進(jìn)行下一步添加hot patch了。
添加hot patch
1、你可以搜索想要的方法名。
為了方便,我們選擇[ViewController viewDidLoad]方法。
- toggleJS是讓我們自己添加js代碼,寫邏輯。
- Before the function有2種操作,一種是彈個alert,可以自己設(shè)置title,msg等。另一種是進(jìn)行事件統(tǒng)計。
- ConditionalPatch是添加patch觸發(fā)的條件,可設(shè)置系統(tǒng)版本,手機(jī)型號。
- 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;
}
);
}
運行結(jié)果如下:
2、Polyfille method
可以理解為方法替代。只能給已存在的類添加方法。可以添加新的方法,或者是給已聲明未實現(xiàn)的方法添加實現(xiàn)。
下面的例子中因為[MyObject test]未實現(xiàn),在MyViewController里面ViewDidload方法調(diào)用了[MyObject test]方法,正常情況下是跳轉(zhuǎn)到MyViewController頁面就會崩潰的。等下用hot patch來解決。經(jīng)測試,如果實現(xiàn)[MyObject test],就會走原有邏輯,patch就不起作用了。如果要修改已存在方法,可參照第三條。
@interface MyObject : NSObject
- (void)say;
// 未實現(xiàn),之后會用hot patch實現(xiàn)
- (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]方法。
然后添加實現(xiàn)。這里我們打開百度頁面。js的寫法也比較簡單。
R.NSClassFromString('UIApplication').call('sharedApplication').call('openURL:', R.NSClassFromString('NSURL').call('URLWithString:', 'http://www.baidu.com'));
調(diào)用方法就用call,傳參數(shù),直接跟在后面。
運行,跳轉(zhuǎn)到MyViewController,會跳轉(zhuǎn)到web。
3、修改已存在的方法
還是拿[ViewController viewDidLoad]來說,在里面調(diào)用方法[MyObject say]。選擇Toggler JS。[MyObject say]是在工程中的方法。
var obj = R.NSClassFromString('MyObject').call('new')
obj.call('say')
self.originalImplementation();
再次運行。
4、修改參數(shù)值
將calculate參數(shù)為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];
}
運行幾次,發(fā)現(xiàn)的確是輸出了20。