本文参考JSPatch wiki :https://github.com/bang590/JSPatch/wiki

1.概念

JSPatch是一个轻量的JS引擎,能够使用JavaScript语言来调用任何object-c接口,替换任何原生的方法。目前主要用于发步JS脚本替换原生Objective-C代码,实时修复线上bug

2.原理

利用OC语言的动态性,动态的修改类的方法和属性。在app启动的时候加载我们写好的JavaScript文件并通过JavaScriptCore来执行,用JS写好的类函数去篡改原有的OC函数。JSPatch只提供了篡改这个过程的代码,像部署线上Js代码、下载这些逻辑都得自己写。当然你可以用JSpatchSDK这个平台,这个平台帮我们部署JS代码、下载等一些逻辑。JSPatchSDK是收费的,也有免费版的。

3.部署

通过pod或者其他方式引入JSPatch,在didFinishLaunchingWithOptions函数加上如下代码:

[JPEngine startEngine];
NSString *sourcePath = [[NSBundle mainBundle] pathForResource:@"index" ofType:@"js"];
NSString *script = [NSString stringWithContentsOfFile:sourcePath encoding:NSUTF8StringEncoding error:nil];
[JPEngine evaluateScript:script];

部署完OC代码,我们就可以在index.js专心写JavaScript来修复线上的bug.

4.写JS代码

  • defineClass 用来定义(覆盖)一个类
  • defineClass(classDeclaration, [properties,] instanceMethods, classMethods)
    
    classDeclaration:字符串类型,代表类名字
    properties:一个字符串数组,代表要添加的属性列表
    instanceMethods:实例方法
    classMethods:累方法
  • 下面是一个例子:
  • 包含了22个知识点,学会了这22个点,就能完成JSPatch的大部分工作了
  • 如果还有不明白的地方可以看官网
  • 1.appDelegate.m文件
  •     [JPEngine startEngine];
    NSString *sourcePath = [[NSBundle mainBundle] pathForResource:@"index" ofType:@"js"];
    NSString *script = [NSString stringWithContentsOfFile:sourcePath encoding:NSUTF8StringEncoding error:nil];
    [JPEngine evaluateScript:script]; self.window = [[UIWindow alloc]initWithFrame:[[UIScreen mainScreen] bounds]]; UINavigationController * navi = [[UINavigationController alloc]initWithRootViewController:[[MainViewController alloc]initWithNibName:@"MainViewController" bundle:nil]];
    self.window.rootViewController = navi;
    [self.window makeKeyAndVisible];
  • 2.MainViewController.h
  • //
    // MainViewController.h
    // JSPatchDemo
    //
    // Created by 朱国清 on 16/12/23.
    // Copyright © 2016年 bang. All rights reserved.
    // #import <UIKit/UIKit.h> @interface MainViewController : UIViewController @property (nonatomic,strong) NSArray * testArray;
    @property (nonatomic,strong) NSDictionary * testDictionary; @end
  • 3.MainViewController.m
  • //
    // MainViewController.m
    // JSPatchDemo
    //
    // Created by 朱国清 on 16/12/23.
    // Copyright © 2016年 bang. All rights reserved.
    // #import "MainViewController.h" @interface MainViewController () @property NSString * privateKey; @end @implementation MainViewController - (void)viewDidLoad {
    [super viewDidLoad]; self.testArray = @[@"Apple",@"Boy",@"Cat",@"Dog",];
    self.testDictionary = @{@"name":@"jack",@"age":@};
    self.privateKey = @"i am a private key"; [self initUI]; }
    // js 覆盖
    -(void)initUI{ if (false) {
    UIButton * btn = [[UIButton alloc]initWithFrame:CGRectMake(,, , )];
    [btn setTitle:@"go" forState:UIControlStateNormal];
    [btn setBackgroundColor:[UIColor brownColor]];
    [btn addTarget:self action:@selector(handleButton) forControlEvents:UIControlEventTouchUpInside]; }
    } -(void)handleButton{ } -(void)testString:(NSString *)string{
    NSLog(@"testString : %@",string);
    }
    - (void)testPointer:(NSError **)error {
    NSError *err = [[NSError alloc]initWithDomain:@"com.jspatch" code: userInfo:nil];
    *error = err;
    }
    -(void)request:(void(^)(NSString *content, BOOL success))callback
    {
    callback(@"I'm content", YES);
    }
    typedef void (^JSBlock)(NSString *str);
    -(JSBlock)getBlock{
    JSBlock block = ^(NSString *str) {
    NSLog(@"I'm %@",str);
    };
    return block;
    } - (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    } @end
  • 4.index.js
  • defineClass(
    'MainViewController',
    {
    viewWillAppear:function(animated){
    // 1.调用父类 self.super()
    self.super().viewWillAppear(animated);
    // 2.设置navigationBarHidden属性值 .setNavigationBarHidden(true)
    self.navigationController().setNavigationBarHidden(true);
    },
    initUI:function(){ // 3.oc的NSString对象,在JS里面是Object
    // 4.privateKey 为私有属性
    var privateKey = self.valueForKey("privateKey");
    self.testString(privateKey);
    console.log('privateKey:'+ privateKey); // 5.OC NSArray
    var item1 = self.testArray().objectAtIndex();
    console.log('item1:'+item1.toString());
    // 6.OC NSArray => JS array
    var jsArray = self.testArray().toJS();
    console.log('jsArray[0]:'+jsArray[]);
    // 7.OC NSDictionary
    var name1 = self.testDictionary().valueForKey('name');
    console.log('name1:'+name1);
    // 8.OC NSDictionary => JS json
    var dic = self.testDictionary().toJS();
    console.log('name2:'+dic['name']); // 10.获取属性 self.view()
    // 11.获取一个类 require('UIColor')
    self.view().setBackgroundColor(require('UIColor').grayColor());
    /*
    12.
    特殊变量
    point {x:, y: }
    size {width:, height:}
    CGRect {x:, y:, width:, height:}
    range {location:, length:}
    */
    var btn = require('UIButton').alloc().initWithFrame({x:, y:, width:, height:});
    btn.setBackgroundColor(UIColor.brownColor());
    // 13.多参数函数 [btn setTitle:forState:];以_代替: btn.setTitle_forState()
    btn.setTitle_forState('go',);
    // 14.Selector 用字符代替
    btn.addTarget_action_forControlEvents(self,'handlePress', << );
    self.view().addSubview(btn); // 15.block无法使用self,需要保存一下
    var slf = self;
    // 16.
    // 弱引用 var weakSelf = __weak(self)
    // 强引用 var strongSelf = __strong(self) // 17. block
    // block block的参数类型用字符串标示,block对象用NSBlock*类型
    /*
    从 JS 传 block 到 OC,有两个限制:
    A. block 参数个数最多支持6个。(若需要支持更多,可以修改源码)
    B. block 参数类型不能是 double / NSBlock / struct 类型。
    */
    self.request(block("NSString *, BOOL", function(ctn, succ) {
    if (succ)
    console.log(ctn) //output: I'm content slf.testString("block self"); }));
    // 18. oc 返回的block就是一个js函数,可以直接调用
    slf.getBlock()('JavaScrict'); // 19.GCD
    // JS
    dispatch_after(1.0, function(){
    console.log('dispatch_after 1s');
    })
    dispatch_async_main(function(){
    console.log('dispatch_async_main');
    })
    dispatch_sync_main(function(){
    console.log('dispatch_sync_main');
    })
    dispatch_async_global_queue(function(){
    console.log('dispatch_async_global_queue');
    }) // 20.参入id * 指针
    //malloc() pval() free() is provided by JPMemory extension
    require('JPEngine').addExtensions(['JPMemory'])
    var pError = malloc(sizeof("id"))
    self.testPointer(pError)
    var error = pval(pError)
    if (!error) {
    console.log("success")
    } else {
    console.log(error)
    }
    releaseTmpObj(pError)
    free(pError)
    },
    handlePress:function(){
    console.log('handlePress'); var vc = InfoViewController.alloc().init();
    self.navigationController().pushViewController_animated(vc,true);
    }
    }
    ) // 21.定义一个继承 UIViewController的类
    // 22.定义两个属性 url、info
    defineClass('InfoViewController : UIViewController',
    [
    'url',
    'info',
    ],
    {
    viewDidLoad:function(){
    self.super().viewDidLoad();
    self.navigationController().setNavigationBarHidden(false);
    self.view().setBackgroundColor(require('UIColor').whiteColor());
    self.setUrl('www.pingan.com.cn');
    console.log(self.url()); } }
    );


JSPatch学习笔记的更多相关文章

  1. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  2. PHP-自定义模板-学习笔记

    1.  开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2.  整体架构图 ...

  3. PHP-会员登录与注册例子解析-学习笔记

    1.开始 最近开始学习李炎恢老师的<PHP第二季度视频>中的“章节5:使用OOP注册会员”,做一个学习笔记,通过绘制基本页面流程和UML类图,来对加深理解. 2.基本页面流程 3.通过UM ...

  4. 2014年暑假c#学习笔记目录

    2014年暑假c#学习笔记 一.C#编程基础 1. c#编程基础之枚举 2. c#编程基础之函数可变参数 3. c#编程基础之字符串基础 4. c#编程基础之字符串函数 5.c#编程基础之ref.ou ...

  5. JAVA GUI编程学习笔记目录

    2014年暑假JAVA GUI编程学习笔记目录 1.JAVA之GUI编程概述 2.JAVA之GUI编程布局 3.JAVA之GUI编程Frame窗口 4.JAVA之GUI编程事件监听机制 5.JAVA之 ...

  6. seaJs学习笔记2 – seaJs组建库的使用

    原文地址:seaJs学习笔记2 – seaJs组建库的使用 我觉得学习新东西并不是会使用它就够了的,会使用仅仅代表你看懂了,理解了,二不代表你深入了,彻悟了它的精髓. 所以不断的学习将是源源不断. 最 ...

  7. CSS学习笔记

    CSS学习笔记 2016年12月15日整理 CSS基础 Chapter1 在console输入escape("宋体") ENTER 就会出现unicode编码 显示"%u ...

  8. HTML学习笔记

    HTML学习笔记 2016年12月15日整理 Chapter1 URL(scheme://host.domain:port/path/filename) scheme: 定义因特网服务的类型,常见的为 ...

  9. DirectX Graphics Infrastructure(DXGI):最佳范例 学习笔记

    今天要学习的这篇文章写的算是比较早的了,大概在DX11时代就写好了,当时龙书11版看得很潦草,并没有注意这篇文章,现在看12,觉得是跳不过去的一篇文章,地址如下: https://msdn.micro ...

随机推荐

  1. hdu 1556

    Color the ball Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  2. MFC获取Windows启动状态(正常启动、安全模式启动、网络安全模式启动)

    UINT nFlags = GetSystemMetrics(SM_CLEANBOOT); switch(nFlags) { : AfxMessageBox(TEXT("正常启动" ...

  3. Java被忽略的基本知识(一)

    工作一段时间发现Java的基本知识不是一般的重要,基本知识好项目理解快.特此看java的基本知识书本<Java项目实战开发>李兴华版,看完此书准备再看一下java更深一点的书--<J ...

  4. LINQ使用细节之.AsEnumerable()和.ToList()的区别

    先看看下面的代码,用了 .AsEnumerable(): 1 var query = (from a in db.Table2 where a = SomeCondition3 select a.So ...

  5. Oracle11完全卸载

    1.停用oracle服务:进入计算机管理,在服务中,找到oracle开头的所有服务,右击选择停止 2.在开始菜单中,找到Universal Installer,运行Oracle Universal I ...

  6. 多Web服务器之间共享Session的解决方案

    一.提出问题: 为了满足足够大的应用,满足更多的客户,于是我们架设了N台Web服务器(N>=2),在多台Web服务器的情况下,我们会涉及到一个问题:用户登陆一台服务器以后,如果在跨越到另一台服务 ...

  7. CF 484E - Sign on Fence

    E. Sign on Fence time limit per test 4 seconds memory limit per test 256 megabytes input standard in ...

  8. Welogic中的JMS

    WebLogic JMS 概述 企业消息传递系统使得应用程序能够通过消息的交换与其他系统之间进行通信.消息是包含协调不同应用程序之间通信所需信息的请求.报告和/或事 件.消息提供了提取级别,使您能够从 ...

  9. MFC编辑框字体大小调节(转)

    在学习MFC中需要调整编辑框中的字体大小,以下是我结合网上与自己实际操作总结的,希望对其它同学有所帮助.       首先,了解以下函数原型:BOOL CreateFont( int nHeight, ...

  10. TOJ 2776 CD Making

    TOJ 2776题目链接http://acm.tju.edu.cn/toj/showp2776.html 这题其实就是考虑的周全性...  贡献了好几次WA, 后来想了半天才知道哪里有遗漏.最大的问题 ...