JSPatch在MAC下的使用

简单调研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 != 3) {
            NSLog(@"usage: testJP [JSPatch.js path] [replace js path]");
            return -1;
        }
        NSString *jsPatchPath = [NSString stringWithUTF8String:argv[1]];
        NSString *replaceJSPath = [NSString stringWithUTF8String:argv[2]];
        
        [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 0;
}

第四步,编写热补丁文件,用于替换第二步中的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文件即可,小巧、便捷,组件化的问题则要靠框架设计来解决。

 

posted @ 2015-11-12 23:55 palance 阅读(...) 评论(...) 编辑 收藏