Cocos2d-x Js Binding 的手动绑定实现

http://www.ityran.com/archives/4902

Cocos2d-x Js Binding 的手动绑定实现

随着 Cocos2d-x 的发展,Cocos2d-html5 也日益完善,相比纯 C++ 的开发方式,它开发效率更为高效,而另一个显而易见的好处便是 JS 端的 API 可以作为 Cocos2d-x Javascript Bindings (JSB) 的接口封装。一套 API,两种解决方案,这让用 JS 快速开发游戏,通过 JSB 以接近原生代码的速度来运行游戏成为可能。

这里使用当前稳定版 Cocos2d-x-2.1.4,Xcode JSB 项目模板创建项目,如果是用其它 IDE ,注意配置好不同环境的依赖关系,本文的示例源码可以在 【这里】 看到。

JSB 手动绑定的实现步骤

要实现 C++ 到 JS 的手动绑定,首先我们需要定义一个待绑定的类,为了这里的解说简单,创建了一个非常简单的类,也只定义了些简单的方法,如下:

   
   
  1. // Leafsoar.h 文件定义
  2. namespace ls {
  3.  class Leafsoar: public cocos2d::CCObject
  4.  {
  5.  public:
  6.  static cocos2d::CCScene* scene();
  7.  virtual bool init();
  8. CREATE_FUNC(Leafsoar);
  9.  void functionTest();
  10.  };
  11. }
  12.  
  13. // Leafsoar.cpp 实现
  14. bool ls::Leafsoar::init(){
  15.  bool bRef = false;
  16.  do {
  17. cocos2d::CCLog("leafsoar init ...");
  18. bRef = true;
  19.  } while (0);
  20.  return bRef;
  21. }
  22. void ls::Leafsoar::functionTest(){
  23. cocos2d::CCLog("function Test");
  24. }

以上是我们定义的一个类,在 ls 命名空间里面,它很简单,继承自 CCObject,定义实现了 functionTest 方法,我们下面要做的就是将它绑定到 JS ,最终达到通过 JS 来创建对象,并且调用方法。如果不知道从何下手,那么下面是一种实现思路。

为了使代码风格统一 (这样的好处是任何人都能相对容易的读懂代码并修改之),我们将参照 Cocos2d-x 现有的 JSB 实现,如从 AppDelegate 的 applicationDidFinishLaunching 方法开始,里面实现了 JSB 环境的初始化等操作,其中我们看到类似 sc->addRegisterCallback(register_all_cocos2dx); 这样的代码,而我们将创建 register_all_ls 方法,来完成我们自有 ls 命名空间下需要绑定的代码。

编写 jsb_ls_auto.h 文件,定义如下:

   
   
  1. #include "jsapi.h"
  2. #include "jsfriendapi.h"
  3. #include "ScriptingCore.h"
  4.  
  5. void register_all_ls(JSContext* cx, JSObject* obj);

完成了以上 register_all_ls 方法定义,它作为自定义 JSB 手动绑定函数的入口,内中实现绑定我么的命名空间,我们的类和方法等 ~ 所以 js_ls_auto.cpp 的实现需要根据自己的需要实现,以下是当前的实现步骤,:

   
   
  1. #include "jsb_ls_auto.h"
  2. #include "cocos2d.h"
  3. #include "Leafsoar.h"
  4.  
  5. #include "cocos2d_specifics.hpp"
  6.  
  7. // 定义 js 端的类型
  8. JSClass *jsb_LsLeafsoar_class;
  9. JSObject *jsb_LsLeafsoar_prototype;
  10.  
  11. // 实现 ls 命名空间下的类绑定
  12. void register_all_ls(JSContext* cx, JSObject* obj) {
  13. jsval nsval;
  14.  JSObject *ns;
  15. JS_GetProperty(cx, obj, "ls", &nsval);
  16.  if (nsval == JSVAL_VOID) {
  17. ns = JS_NewObject(cx, NULL, NULL, NULL);
  18. nsval = OBJECT_TO_JSVAL(ns);
  19. JS_SetProperty(cx, obj, "ls", &nsval);
  20.  } else {
  21. JS_ValueToObject(cx, nsval, &ns);
  22.  }
  23. obj = ns;
  24.  
  25.  // 实现绑定 Leafsoar 类,它的定义后文给出
  26. js_register_ls_Leafsoar(cx, obj);
  27. }

为了实现思路的清晰,所以文章内容以 register_all_ls 为入口,一步步实现,需要什么,我们就去实现什么,看到上面绑定了命名空间(在 js 中并没有明确的命名空间的机制,但 js 能实现类似命名空间的效果),并调用了js_register_ls_Leafsoar(cx, obj); 方法来实现具体的绑定,下面是它的实现:

   
   
  1. // 绑定 Leafsoar 类的实现
  2. void js_register_ls_Leafsoar(JSContext *cx, JSObject *global) {
  3.  // 创建一个 JS 类型的对象
  4. jsb_LsLeafsoar_class = (JSClass *)calloc(1, sizeof(JSClass));
  5.  // 类型名称为 **Leafsoar** 正式绑定到 js 由 js 调用的名称
  6. jsb_LsLeafsoar_class->name = "Leafsoar";
  7. jsb_LsLeafsoar_class->addProperty = JS_PropertyStub;
  8. jsb_LsLeafsoar_class->delProperty = JS_PropertyStub;
  9. jsb_LsLeafsoar_class->getProperty = JS_PropertyStub;
  10. jsb_LsLeafsoar_class->setProperty = JS_StrictPropertyStub;
  11. jsb_LsLeafsoar_class->enumerate = JS_EnumerateStub;
  12. jsb_LsLeafsoar_class->resolve = JS_ResolveStub;
  13. jsb_LsLeafsoar_class->convert = JS_ConvertStub;
  14.  // Leafsoar 类型的析构函数绑定
  15. jsb_LsLeafsoar_class->finalize = js_ls_Leafsoar_finalize;
  16. jsb_LsLeafsoar_class->flags = JSCLASS_HAS_RESERVED_SLOTS(2);
  17.  
  18.  static JSPropertySpec properties[] = {
  19.  {0, 0, 0, JSOP_NULLWRAPPER, JSOP_NULLWRAPPER}
  20.  };
  21.  
  22.  // 为 Leafsoar 设定绑定函数,函数名 "functionTest",绑定函数 "js_ls_Leafsoar_functionTest"
  23.  // 后面可以添加其它函数绑定,如果需要,之后以 "JS_FS_END" 结尾
  24.  static JSFunctionSpec funcs[] = {
  25. JS_FN("functionTest", js_ls_Leafsoar_functionTest, 1, JSPROP_PERMANENT | JSPROP_ENUMERATE),
  26. JS_FS_END
  27.  };
  28.  
  29.  // 这里定义并且绑定了静态函数(static),包括方法名 "create" 和对应的绑定实现 "js_ls_Leafsoar_create"
  30.  static JSFunctionSpec st_funcs[] = {
  31. JS_FN("create", js_ls_Leafsoar_create, 0, JSPROP_PERMANENT | JSPROP_ENUMERATE),
  32. JS_FS_END
  33.  };
  34.  
  35.  // 初始化类型属性
  36. jsb_LsLeafsoar_prototype = JS_InitClass(
  37. cx, global,
  38. NULL, // parent proto
  39. jsb_LsLeafsoar_class,
  40. js_ls_Leafsoar_constructor, 0, // 这里绑定的是构造函数的实现,也就是用 js new 操作符创建的对象
  41. properties,
  42. funcs, // 函数绑定
  43. NULL, // no static properties
  44. st_funcs); // 静态函数绑定
  45.  
  46.  JSBool found;
  47. JS_SetPropertyAttributes(cx, global, "Leafsoar", JSPROP_ENUMERATE | JSPROP_READONLY, &found);
  48.  
  49.  TypeTest<ls::Leafsoar> t;
  50.  js_type_class_t *p;
  51.  uint32_t typeId = t.s_id();
  52. HASH_FIND_INT(_js_global_type_ht, &typeId, p);
  53.  if (!p) {
  54. = (js_type_class_t *)malloc(sizeof(js_type_class_t));
  55. p->type = typeId;
  56. p->jsclass = jsb_LsLeafsoar_class;
  57. p->proto = jsb_LsLeafsoar_prototype;
  58. p->parentProto = NULL;
  59. HASH_ADD_INT(_js_global_type_ht, type, p);
  60.  }
  61. }

写到这里,类型的绑定已经基本完成,但是可以看见,其中所用到的如 js_ls_Leafsoar_functionTestjs_ls_Leafsoar_finalize 、js_ls_Leafsoar_create 和 js_ls_Leafsoar_constructor 并没有实现,它们是在绑定 Leafosar 类型的时候去绑定了,所以需要在调用前去实现它们,下面是它们的实现:

   
   
  1. // js 端 functionTest 所绑定的方法调用
  2. JSBool js_ls_Leafsoar_functionTest(JSContext *cx, uint32_t argc, jsval *vp)
  3. {
  4.  JSBool ok = JS_TRUE;
  5.  
  6.  JSObject *obj = NULL;
  7. ls::Leafsoar* cobj = NULL; // 定义以获取真实类型
  8. obj = JS_THIS_OBJECT(cx, vp);
  9.  js_proxy_t *proxy = jsb_get_js_proxy(obj);
  10.  // 获取 js 绑定的实际对象 通过 proxy->ptr
  11. cobj = (ls::Leafsoar *)(proxy ? proxy->ptr : NULL);
  12. JSB_PRECONDITION2( cobj, cx, JS_FALSE, "Invalid Native Object");
  13.  if (argc == 0) {
  14.  // 调用实际的方法
  15. cobj->functionTest();
  16. JS_SET_RVAL(cx, vp, JSVAL_VOID);
  17.  return ok;
  18.  }
  19.  
  20. JS_ReportError(cx, "wrong number of arguments");
  21.  return JS_FALSE;
  22. }
  23.  
  24. // js 构造函数实现
  25. JSBool js_ls_Leafsoar_constructor(JSContext *cx, uint32_t argc, jsval *vp)
  26. {
  27. cocos2d::CCLog("js ls lsleafsoar constructor ..");
  28.  if (argc == 0) {
  29.  // 调用 C++ 构造函数
  30. ls::Leafsoar* cobj = new ls::Leafsoar();
  31. cocos2d::CCObject* _ccobj = dynamic_cast<cocos2d::CCObject*>(cobj);
  32.  // 默认使用原有的内存管理方式
  33.  if (_ccobj){
  34. _ccobj->autorelease();
  35.  }
  36.  TypeTest<ls::Leafsoar> t;
  37.  js_type_class_t *typeClass;
  38.  uint32_t typeId = t.s_id();
  39. HASH_FIND_INT(_js_global_type_ht, &typeId, typeClass);
  40.  assert(typeClass);
  41.  JSObject *obj = JS_NewObject(cx, typeClass->jsclass, typeClass->proto, typeClass->parentProto);
  42. JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
  43.  // 构造 js 端对象,将 cobj 实际对象存入
  44.  js_proxy_t* p = jsb_new_proxy(cobj, obj);
  45. JS_AddNamedObjectRoot(cx, &p->obj, "ls::Leafsoar");
  46.  
  47.  return JS_TRUE;
  48.  }
  49.  
  50. JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 0);
  51.  return JS_FALSE;
  52. }
  53.  
  54. // 静态函数 create 的具体实现
  55. JSBool js_ls_Leafsoar_create(JSContext *cx, uint32_t argc, jsval *vp)
  56. {
  57. cocos2d::CCLog("js ls lsleafsoar create ..");
  58.  if (argc == 0) {
  59.  // 创建 Leafsoar 对象
  60. ls::Leafsoar* ret = ls::Leafsoar::create();
  61. jsval jsret;
  62.  do {
  63.  if (ret) {
  64.  js_proxy_t *proxy = js_get_or_create_proxy<ls::Leafsoar>(cx, ret);
  65. jsret = OBJECT_TO_JSVAL(proxy->obj);
  66.  } else {
  67. jsret = JSVAL_NULL;
  68.  }
  69.  } while (0);
  70. JS_SET_RVAL(cx, vp, jsret);
  71.  return JS_TRUE;
  72.  }
  73. JS_ReportError(cx, "wrong number of arguments");
  74.  return JS_FALSE;
  75. }
  76.  
  77. void js_ls_Leafsoar_finalize(JSFreeOp *fop, JSObject *obj) {
  78.  // 析构函数实现,如果在构造函数做了什么,如开辟内存空间,那么需要在这里做些收尾工作
  79.  // CCLOGINFO("jsbindings: finalizing JS object %p (LsLeafsoar)", obj);
  80. }

通过以上的步骤,我们实现了 C++ 类 Leafosar 到 JS 端的绑定。在 JS 中我们可以通过以下调试测试:

   
   
  1. // var ls = new ls.Leafsoar();
  2. // 或者
  3. var ls = ls.Leafsoar.create();
  4. // 之后调用
  5. ls.functionTest();

怎样实现 C++ 回调 JS

在上文,完成了 C++ 到 js 的手动绑定,但有时我们还需要其它一些功能,比如想在 C++ 开一个多线程以加载资源,或者一个网络异步请求,再如要实现一个 delegate 以实现接口回调,然这些都归为同一个问题,实现 C++ 到 js 的回调。我们在 js 端定义了一个 Leafsoar 对象,并且新实现了一个方法,等待 C++ 端的回调,如下:

   
   
  1. var ls = new ls.Leafsoar(); // 创建一个对象
  2. // 定义回调函数 callback
  3. ls.callback = function(i, j){
  4. log("ls.callback " + i + j);
  5. };
  6. ls.functionTest();

我们想通过调用 functionTest 之后回调在 js 端定义的 callback 方法。那么我们需要重新实现 C++ 端的 functionTest 方法:

   
   
  1. void ls::Leafsoar::functionTest(){
  2. cocos2d::CCLog("function Test");
  3.  js_proxy_t * p = jsb_get_native_proxy(this);
  4. jsval retval;
  5.  JSContext* jc = ScriptingCore::getInstance()->getGlobalContext();
  6.  // 定义参数,由两个参数
  7. jsval v[] = {
  8. v[0] = int32_to_jsval(jc, 32),
  9. v[1] =UINT_TO_JSVAL(88)
  10.  };
  11.  // 通过 ScriptingCore 封装好的方法实现回调,可以帮助我们节省很多细节上的研究
  12.  ScriptingCore::getInstance()->executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "callback", 2, v, &retval);
  13. }

JSB 的内存管理

了解 Cocos2d-x 的朋友知道,它的内存管理方式,如果对此有疑问,可以参见 Cocos2d-x 内存管理浅说 和 深入理解 Cocos2d-x 内存管理 这两篇文章,那么在 JSB 我们如何来管理内存呢?在 C++ 需要通过 retain 和 release 来实现引用计数的管理(源码示例也给出它的绑定实现,但仅仅作为参考),在绑定 js 时,如果不做相应处理,那么可能会出现 js 正在运行着的代码,所绑定的实际 C++ 对象已经释放。虽然我们能通过 绑定实现 retain 和 release 方法,来实现 js 端的此方法调用,但这显然不符合 js 代码边的习惯,它是自动回收的,所以这里推荐 始终 由 SpiderMonkey 来保持一份对象引用,以使它更像 JS 的使用方式,当 js 垃圾回收自动执行时,在去释放 SpiderMonkey 对对象的引用。

要做到这一点,我们需要只要修改上文的代码实现,在 构造函数,create 静态方法,实现对 C++ 类型对象的引用,在 析构绑定的析构函数中解除对其的引用以完成 C++ 到 JS 端绑定的内存管理方案。

本文出自:http://www.ityran.com/archives/4902

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值