简单调研JSPatch的使用,之所以在MAC下是因为可以创建一个命令行的应用,简化无关代码。具体做法如下:
第一步,去https://github.com/bang590/JSPatch/tree/master/JSPatch下载JSEngine.h、JSEngine.m和JSPatch.js三个文件,并拖入工程,并选择Copy items if needed:
第二步,添加一个被测试的类,并给它定义一个类方法,稍后我们将用JSPatch替换该方法。我在该方法中输出了字符串:Hello World!
// TestObj.h
#import <Foundation/Foundation.h> @interface TestObj : NSObject
-(void)mainProc;
@end
//TestObj.m
#import "TestObj.h" @implementation TestObj
-(void)mainProc
{
NSLog(@"Hello World!");
}
@end
第三步,在main函数中接收两个参数,一个是在第一步中下载的JSPatch.js,另一个是用来做热补丁的js文件,命名为demo.js
紧接着就可以执行JSPatch的策略了,之后才是业务逻辑。必须在执行业务逻辑之前执行JSPatch策略,这样才能让对业务逻辑中的类或方法的替换生效。
#import <Foundation/Foundation.h>
#import "JPEngine.h"
#import "TestObj.h" int main(int argc, const char * argv[]) {
@autoreleasepool {
// 传入JSPatch.js和demo.js的路径
if (argc != ) {
NSLog(@"usage: testJP [JSPatch.js path] [replace js path]");
return -;
}
NSString *jsPatchPath = [NSString stringWithUTF8String:argv[]];
NSString *replaceJSPath = [NSString stringWithUTF8String:argv[]]; [JPEngine startEngine:jsPatchPath];
NSString *script = [NSString stringWithContentsOfFile:replaceJSPath encoding:NSUTF8StringEncoding error:nil];
[JPEngine evaluateScript:script]; // 实例化TestObj,并调用将被JSPath替换的方法
TestObj *obj = [[TestObj alloc]init];
[obj mainProc];
}
return ;
}
第四步,编写热补丁文件,用于替换第二步中的TestObj的mainProc方法:
// demo.js
// 替换TestObj的mainProc方法
defineClass('TestObj', {
mainProc: function() {
console.log("Hello, JSPatch!");
}
});
以上就是应用JSPatch替换掉OC代码中TestObj的mainPrco方法的步骤,不过要想在MAC Command Line下能跑,还得对JPEngine稍作修改:
1、JPEngine依赖了UIKit/UIApplication.h,这是应用于iOS UIKit的头文件,可以直接注掉。
2、在startEngine函数中有如下一段调用,这也是在iOS中用于响应内存警告的,也可以直接注掉。
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleMemoryWarning) name:UIApplicationDidReceiveMemoryWarningNotification object:nil];
3、紧接着往下是如下一段代码,这是直接在bundle中查找JSPatch.js,而我们创建的是命令行应用而不是app,尽管在第一步中把JSPatch.js拖入了工程,通过下面的方法也是拿不到JSPatch.js的,
NSString *path = [[NSBundle bundleForClass:[self class]] pathForResource:@"JSPatch" ofType:@"js"];
NSAssert(path, @"can't find JSPatch.js");
NSString *jsCore = [[NSString alloc] initWithData:[[NSFileManager defaultManager] contentsAtPath:path] encoding:NSUTF8StringEncoding];
因此,我把JPEngine的startEngine方法由无参改成接收一个NSSting*参数,通过命令行第一个参数把JSPatch.js的路径传进来,前面这段代码可以改为如下,jsPatchPatch是startEngine的参数
//*#*# 注掉path,改为传参
// NSString *path = [[NSBundle bundleForClass:[self class]] pathForResource:@"JSPatch" ofType:@"js"];
// NSAssert(path, @"can't find JSPatch.js");
NSString *jsCore = [[NSString alloc] initWithData:[[NSFileManager defaultManager] contentsAtPath:jsPatchPath] encoding:NSUTF8StringEncoding];
经过以上改造,JSPatch就可以在MAC下使用了。实例代码可在https://github.com/palanceli/TestJSPatch/tree/master下载。
本文只是简单摸索一下JSPatch的使用,并没有深入到JSPatch内部,据说这种热补丁覆盖面太广会让效率不彰,因为它是利用字符串反射找到对应类和方法的。我认为JSPatch的应用目标就是用来打补丁而不是用于组件化,临时遇到bug,需要紧急修复还是很适合用JSPatch的,下发一个js文件即可,小巧、便捷,组件化的问题则要靠框架设计来解决。