zoukankan      html  css  js  c++  java
  • 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文件即可,小巧、便捷,组件化的问题则要靠框架设计来解决。

  • 相关阅读:
    Java内存泄漏的排查总结
    多文本匹配 AC算法(Aho-Corasick)
    Linux下进程管理利器—supervise(监控并将死掉的程序重启)
    maven pom项目的dependencies转gradle格式
    Protocol Buffers官方文档(开发指南)
    Spring的工具类StringUtils使用
    springboot2 中Druid和ibatis(baomidou) 遇到org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): com.iflytek.pandaai.service.multi.mapper.TanancyMapper
    gradle 多模块Springboot项目 compile project引用其他模块的坑
    JAVA 运行springboot jar包设置classpath
    mysql8
  • 原文地址:https://www.cnblogs.com/palance/p/4960684.html
Copyright © 2011-2022 走看看