博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
JSPatch库, 一个Apple官方支持的实现在线更新iOS应用的库
阅读量:5115 次
发布时间:2019-06-13

本文共 6333 字,大约阅读时间需要 21 分钟。

简介

  • 项目主页:
  • 示例下载:

JSPatch 可以让你用 JavaScript 书写原生 iOS APP。只需在项目引入极小的引擎,就可以使用 JavaScript 调用任何 Objective-C 的原生接口,获得脚本语言的优势:为项目动态添加模块,或替换项目原生代码动态修复 bug。

优势

  • 在项目中引入JSPatch,就可以在发现bug时下发JS脚本替换原生方法,可以做到无需更新整个APP即时修复bug!

  • JSPatch用iOS内置的 JavaScriptCore.framework作为引擎;JSPatch也符合苹果的规则。苹果不允许动态下发可执行代码,但通过苹果 JavaScriptCore.framework 或 WebKit 执行的代码除外,JS 正是通过 JavaScriptCore.framework 执行的。

  • JSPatch非常小巧

实例预览

@implementation AppDelegate- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {    [JPEngine startEngine];    NSString *sourcePath = [[NSBundle mainBundle] pathForResource:@"demo" ofType:@"js"];    NSString *script = [NSString stringWithContentsOfFile:sourcePath encoding:NSUTF8StringEncoding error:nil];    [JPEngine evaluateScript:script];        self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];    [self.window addSubview:[self genView]];    [self.window makeKeyAndVisible];        return YES;}- (UIView *)genView{    return [[UIView alloc] initWithFrame:CGRectMake(0, 0, 320, 320)];}@end
// demo.jsrequire('UIView, UIColor, UILabel')defineClass('AppDelegate', {  // 替换这个 -genView 方法  genView: function() {    var view = self.ORIGgenView();    view.setBackgroundColor(UIColor.greenColor())    var label = UILabel.alloc().initWithFrame(view.frame());    label.setText("JSPatch");    label.setTextAlignment(1);    view.addSubview(label);    return view;  }});

安装

通过Cocopods安装

pod 'JSPatch' # 在线更新应用.

手动导入

  1. 下载并解压

  2. 复制JSPatch文件夹到你的工程

使用

objective-C:

  1. 导入头文件#import "JPEngine.h"

  2. 导入本地JS(demo.js)见文首github示例demo(可选,实际项目中,根据自己实际需要进行.)

  3. 调用[JPEngine startEngine] 加载引擎
  4. 通过[JPEngine evaluateScript:@""]接口执行 JavaScript。

[JPEngine startEngine];// 直接执行js[JPEngine evaluateScript:@"\ var alertView = require('UIAlertView').alloc().init();\ alertView.setTitle('Alert');\ alertView.setMessage('AlertView from js'); \ alertView.addButtonWithTitle('OK');\ alertView.show(); \"];// 从网络拉回js脚本执行[NSURLConnection sendAsynchronousRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://cnbang.net/test.js"]] queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {    NSString *script = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];    [JPEngine evaluateScript:script];}];// 执行本地js文件NSString *sourcePath = [[NSBundle mainBundle] pathForResource:@"sample" ofType:@"js"];NSString *script = [NSString stringWithContentsOfFile:sourcePath encoding:NSUTF8StringEncoding error:nil];[JPEngine evaluateScript:script];
// 另一个例子// 加载引擎[JPEngine startEngine];    //  本地JS,动态更新技术就是通过服务器获取JS更新这个JSNSString *sourcePath = [[NSBundle mainBundle] pathForResource:@"demo" ofType:@"js"];NSString *script = [NSString stringWithContentsOfFile:sourcePath encoding:NSUTF8StringEncoding error:nil];[JPEngine evaluateScript:script]

JavaScript:

基础使用方式

// 调用require引入要使用的OC类require('UIView, UIColor, UISlider, NSIndexPath')// 调用类方法var redColor = UIColor.redColor();// 调用实例方法var view = UIView.alloc().init();view.setNeedsLayout();// set proertyview.setBackgroundColor(redColor);// get property var bgColor = view.backgroundColor();// 多参数方法名用'_'隔开:// OC:NSIndexPath *indexPath = [NSIndexPath indexPathForRow:0 inSection:1];var indexPath = NSIndexPath.indexPathForRow_inSection(0, 1);// 方法名包含下划线'_',js用双下划线表示// OC: [JPObject _privateMethod];JPObject.__privateMethod()// 如果要把 `NSArray` / `NSString` / `NSDictionary` 转为对应的 JS 类型,使用 `.toJS()` 接口.var arr = require('NSMutableArray').alloc().init()arr.addObject("JS")jsArr = arr.toJS()console.log(jsArr.push("Patch").join(''))  //output: JSPatch// 在JS用字典的方式表示 CGRect / CGSize / CGPoint / NSRangevar view = UIView.alloc().initWithFrame({x:20, y:20, width:100, height:100});var x = view.bounds.x;// block 从 JavaScript 传入 Objective-C 时,需要写上每个参数的类型。// OC Method: + (void)request:(void(^)(NSString *content, BOOL success))callbackrequire('JPObject').request(block("NSString *, BOOL", function(ctn, succ) {  if (succ) log(ctn)}));// GCDdispatch_after(function(1.0, function(){  // do something}))dispatch_async_main(function(){  // do something})

详细文档请参考wiki页面:

定义类/替换方法

defineClass() 定义 Objective-C 的类,对类和实例方法进行动态替换。

// OC@implementation JPTableViewController...- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{  NSString *content = self.dataSource[[indexPath row]];  //may cause out of bound  JPViewController *ctrl = [[JPViewController alloc] initWithContent:content];  [self.navigationController pushViewController:ctrl];}- (NSArray *)dataSource{  return @[@"JSPatch", @"is"];}- (void)customMethod{  NSLog(@"callCustom method")}@end
// JSdefineClass("JPTableViewController", {  // instance method definitions  tableView_didSelectRowAtIndexPath: function(tableView, indexPath) {    var row = indexPath.row()    if (self.dataSource().count() > row) {  //fix the out of bound bug here      var content = self.dataSource().objectAtIndex(row);      var ctrl = JPViewController.alloc().initWithContent(content);      self.navigationController().pushViewController(ctrl);    }  },  dataSource: function() {    // get the original method by adding prefix 'ORIG'    var data = self.ORIGdataSource().toJS();    return data.push('Good!');  }}, {})

详细文档请参考wiki页面:

扩展

一些自定义的struct类型、C函数调用以及其他功能可以通过扩展实现,调用 +addExtensions: 可以加载扩展接口:

@implementation AppDelegate- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {    [JPEngine startEngine];    //添加扩展    [JPEngine addExtensions:@[@"JPInclude", @"JPCGTransform"]];    NSString *sourcePath = [[NSBundle mainBundle] pathForResource:@"demo" ofType:@"js"];    NSString *script = [NSString stringWithContentsOfFile:sourcePath encoding:NSUTF8StringEncoding error:nil];    [JPEngine evaluateScript:script];}
include('test.js')   //`include()`方法在扩展 JPInclude.m 里提供var view = require('UIView').alloc().init()//struct CGAffineTransform 类型在 JPCGTransform.m 里提供支持view.setTransform({a:1, b:0, c:0, d:1, tx:0, ty:100})

扩展可以在JS动态加载,更推荐这种加载方式,在需要用到时才加载:

require('JPEngine').addExtensions(['JPInclude', 'JPCGTransform'])// `include()` and `CGAffineTransform` is avaliable now.

可以通过新增扩展为自己项目里的 struct 类型以及C函数添加支持,详情请见wiki页面:

安全性

JSPatch非常强大,因而最好将通过服务器获取JS的链接进行加密,本地JS也最好加密处理


注: 文章由我们 iOS122 的小伙伴 @偌一茗 整理,喜欢就一起参与:

转载于:https://www.cnblogs.com/ios122/p/4952112.html

你可能感兴趣的文章
CMDB学习之六 --客户端请求测试,服务端api优化
查看>>
java学习笔记_GUI(4)
查看>>
Vue 页面15分钟无操作时返回首页
查看>>
WinForms 2015V2版本Reports支持更多种条形码!
查看>>
Mysql基本操作总结
查看>>
Xcode 6制作动态及静态Framework
查看>>
经典算法实例:2
查看>>
SpringCloud的DataRest(三)
查看>>
VxWorks 7 技术简介
查看>>
bouncy castle的配置
查看>>
grails-shiro权限认证
查看>>
UITableView 小记
查看>>
用DotNetDetour HOOK .net类库
查看>>
配置文件刷新ConfigurationManager.RefreshSection使用问题
查看>>
设计模式分类
查看>>
Chapter5:语句
查看>>
利用Maven打包时,如何包含更多的资源文件
查看>>
CSRF攻击与防御
查看>>
C# 历史曲线控件 基于时间的曲线控件 可交互的高级曲线控件 HslControls曲线控件使用教程...
查看>>
golang gc
查看>>