JSPatch 可以让你用 JavaScript 书写原生 iOS APP。只需在项目引入极小的引擎,就可以使用 JavaScript 调用任何 Objective-C 的原生接口,替换项目原生代码动态修复 bug。
1.接入流程
JSPatch github地址:https://github.com/bang590/JSPatch
- 拷贝
JSPatch/目录下的三个文件JSEngine.m/JSEngine.h/JSPatch.js到项目。// JSpatch 核心类 - 拷贝
Loader文件夹到项目。// JSpatch 安全管理类 - 拷贝
Extensions/目录下的JPCleaner.h/JPCleaner.m文件到项目。// JSpatch 扩展的清除js的类 - 在
General的LinkFrameworks and Libraries里面 添加javascriptcore.framework//这个库里主要用于js与oc语言的桥接,比如一些数据类型间的相互转化。
2.本地测试
编写main.js文件
- 用原生代码编写修复bug的代码,之后可以使用
JSPatch Convertor自动把 Objective-C 代码转为 JavaScript 代码,然后把生成代码写入main.js文件中,根据作者给的js语法进行一行行review。
- 导入
JSPactch X插件手写js代码。
测试main.js文件
- 把
main.js文件导入项目Resource/目录下。 - 在
didFinishLaunchingWithOptions:方法中调用[JPLoader runTestScriptInBundle];方法。 - 如果bug未修复需要在
main.js中进行断点调试或者使用console.log("xxx");进行调试直到bug修复完成。
断点调试说明

- 首先需要开启
Safari调试菜单:Safari->偏好设置->高级-> 勾选在菜单栏中显示“开发”菜单 - 接着启动APP ->
Safari->开发-> 选择你的机器 ->JSContext即可开始调试。 - 连接真机调试时,需要打开真机的web检查器:
设置->Safari->高级->Web检查器。
3.生成RSA密钥
命令行依次写入
openssl genrsa -out rsatest_private_key.pem 1024pkcs8 -topk8 -inform PEM -in rsatest_private_key.pem -outform PEM –nocryptrsa -in rsatest_private_key.pem -pubout -out rsatest_public_key.pem
会在本地当前用户的目录下生成连个文件如下:

配置
- 文本形式打开
rsatest_public_key.pem替换JPLoader.h里的publicKey。 - 打开
rsatest_private_key.pem替换tools/pack.php里的privateKey。 - 设
JPLoader.h的rootUrl为你的服务器地址。
脚本打包
- 取出
tools内含pack.php的文件夹放在桌面(其他地方也行)。 - 把测试成功的
main.js文件放入tools文件夹。 - 通过命令行
cd命令到此文件夹。 - 敲入命令
packer.php main.js,文件夹下会生成一个v1.zip的包。 - 脚本打包后的文件存放在服务器的路径:
${rootUrl}/${appVersion}/${patchFile}。
4.后台请求测试
- 在
didFinishLaunchingWithOptions:方法中调用[JPLoader run];方法。 - 在
applicationDidBecomeActive:方法中调用[JPLoader updateToVersion:1 callback:nil];方法。
5.添加容错
崩溃x次 清除本地js文件 取消加载
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
NSInteger crashInteger = [[NSUserDefaults standardUserDefaults] integerForKey:@"loadPatchCrash"];
if (crashInteger < 2) { // 加载patch 崩溃2次 取消加载
crashInteger++;
[[NSUserDefaults standardUserDefaults] setInteger:crashInteger forKey:@"loadPatchCrash"];
[[NSUserDefaults standardUserDefaults] synchronize];
[JPLoader run];
crashInteger--;
[[NSUserDefaults standardUserDefaults] setInteger:crashInteger forKey:@"loadPatchCrash"];
[[NSUserDefaults standardUserDefaults] synchronize];
}else {
// 2次 删除本地 patch 文件
[JPCleaner cleanAll];
[[NSUserDefaults standardUserDefaults] setInteger:0 forKey:@"loadPatchCrash"];
[[NSUserDefaults standardUserDefaults] synchronize];
}
}
间隔xxx秒去请求脚步(或者后台返回脚本的版本号判断缓存)
- (void)applicationDidBecomeActive:(UIApplication *)application
{
// JSPatch请求时间间隔的判断
NSDate *lastRequestTime = [[NSUserDefaults standardUserDefaults] objectForKey:@"jp_request_time"];
//下载补丁包得请求
void(^timeInterval)() = ^ {
[JPLoader updateToVersion:1 callback:nil];
// 将当前时间写入沙盒
[[NSUserDefaults standardUserDefaults] setObject:[NSDate date] forKey:@"jp_request_time"];
[[NSUserDefaults standardUserDefaults] synchronize];
};
// 判断
if (lastRequestTime) {
NSTimeInterval between = [lastRequestTime timeIntervalSinceNow];
if (between > 3600) {
timeInterval();
}else return;
}else {
timeInterval();
}
}
