【cocos2d-x从c++到js】15:傀儡构造函数

简介:

上篇我们以Sprite为例,分析了注册函数。但其中我们似乎遗漏了一个地方,那就是构造函数。因为Cocos2d-x在C++层使用的是工场函数来生成对象,而不是构造函数。所以在JS层代码中,也需要有相应的对应机制来处理这件事。


看一下jsb_cocos2dx_auto.hpp

1
2
3
4
5
6
extern  JSClass  *jsb_cocos2d_Sprite_class;
extern  JSObject *jsb_cocos2d_Sprite_prototype;
JSBool js_cocos2dx_Sprite_constructor(JSContext *cx, uint32_t argc, jsval *vp);
void  js_cocos2dx_Sprite_finalize(JSContext *cx, JSObject *obj);
void  js_register_cocos2dx_Sprite(JSContext *cx, JSObject *global);
void  register_all_cocos2dx(JSContext* cx, JSObject* obj);


这声明了几个重要的对象和函数。JSClass对象和原型对象、注册函数、自己实现的finalize的Stub等。但是我们发现js_cocos2dx_Sprite_constructor构造函数并没有对应的实现代码,仅仅是一个声明而已。


需要注意的是,根据JS的原型继承,我们在生成jsb_cocos2d_Sprite_prototype原型时,需要传入一个构造函数,而构造函数js_cocos2dx_Sprite_constructor又是未实现的,那么他是如何做到的呢?


在js_register_cocos2dx_Sprite函数中查看生成jsb_cocos2d_Sprite_prototype原型的代码:

1
2
3
4
5
6
7
8
9
jsb_cocos2d_Sprite_prototype = JS_InitClass(
     cx, global,
     jsb_cocos2d_Node_prototype,
     jsb_cocos2d_Sprite_class,
     dummy_constructor<cocos2d::Sprite>, 0,  // no constructor
     properties,
     funcs,
     NULL,  // no static properties
     st_funcs);

注意到第五个参数是一个模板函数dummy_constructor<cocos2d::Sprite>,字面意思是傀儡构造函数。


看一下这个模板函数的定义

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
template < class  T>
static  JSBool dummy_constructor(JSContext *cx, uint32_t argc, jsval *vp) {
     JS::RootedValue initializing(cx);
     JSBool isNewValid = JS_TRUE;
     JSObject* global = ScriptingCore::getInstance()->getGlobalObject();
     isNewValid = JS_GetProperty(cx, global,  "initializing" , &initializing) && JSVAL_TO_BOOLEAN(initializing);
     if  (isNewValid)
     {
         TypeTest<T> t;
         js_type_class_t *typeClass = nullptr;
         std::string typeName = t.s_name();
         auto  typeMapIter = _js_global_type_map.find(typeName);
         CCASSERT(typeMapIter != _js_global_type_map.end(),  "Can't find the class type!" );
         typeClass = typeMapIter->second;
         CCASSERT(typeClass,  "The value is null." );
         JSObject *_tmp = JS_NewObject(cx, typeClass->jsclass, typeClass->proto, typeClass->parentProto);
         JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(_tmp));
         return  JS_TRUE;
     }
     JS_ReportError(cx,  "Don't use `new cc.XXX`, please use `cc.XXX.create` instead! " );
     return  JS_FALSE;
}

这个函数首先使用了JS::RootedValue类型的量来判断GlobalObject对象是否初始化完毕。JS::RootedValue具体的原理暂时不用深究,你只需要知道这是SpiderMonkey引擎的一种内存管理方式即可。


然后使用了一个非常有趣的技巧,用一个模板类TypeTest<T> t,取出对应的类型名。这是一个很不错的写法,能够不破坏函数签名,使得函数能够匹配JS_InitClass的参数类型,又能够在不同的上下文中里面获得需要的信息。我们看一下TypeTest的实现,这种写法在很多时候有很大的借鉴意义!

1
2
3
4
5
6
7
8
9
10
11
12
template typename  DERIVED >
class  TypeTest
{
public :
     static  const  char * s_name()
     {
         // return id unique for DERIVED
         // ALWAYS VALID BUT STRING, NOT INT - BUT VALID AND CROSS-PLATFORM/CROSS-VERSION COMPATBLE
         // AS FAR AS YOU KEEP THE CLASS NAME
         return  typeid ( DERIVED ).name();
     }
};


最后我们在_js_global_type_map里查询对应的类型,取出相应的参数来调用JS_NewObject函数,生成对应的对象并设置为返回值。





 本文转自 老G 51CTO博客,原文链接:http://blog.51cto.com/goldlion/1357886,如需转载请自行联系原作者


相关文章
|
1月前
|
设计模式 JavaScript 前端开发
在JavaScript中,继承是一个重要的概念,它允许我们基于现有的类(或构造函数)创建新的类
【6月更文挑战第15天】JavaScript继承促进代码复用与扩展,创建类层次结构,但过深的继承链导致复杂性增加,紧密耦合增加维护成本,单继承限制灵活性,方法覆盖可能隐藏父类功能,且可能影响性能。设计时需谨慎权衡并考虑使用组合等替代方案。
37 7
|
1月前
|
编解码 C++
cocos2d-x C++ 工程初探
cocos2d-x C++ 工程初探
|
25天前
|
设计模式 JavaScript 前端开发
js设计模式【详解】—— 构造函数模式
js设计模式【详解】—— 构造函数模式
21 6
|
1月前
|
安全 编译器 C++
C++一分钟之-构造函数与析构函数
【6月更文挑战第20天】C++中的构造函数初始化对象,析构函数负责资源清理。构造函数有默认、参数化和拷贝形式,需注意异常安全和成员初始化。析构确保资源释放,避免内存泄漏,要防止重复析构。示例代码展示了不同构造函数和析构函数的调用情况。掌握构造和析构是有效管理对象生命周期和资源的关键。
28 2
|
25天前
|
编译器 C++
【C++】详解构造函数
【C++】详解构造函数
|
28天前
|
存储 编译器 C++
【C++】类和对象④(再谈构造函数:初始化列表,隐式类型转换,缺省值
C++中的隐式类型转换在变量赋值和函数调用中常见,如`double`转`int`。取引用时,须用`const`以防修改临时变量,如`const int& b = a;`。类可以有隐式单参构造,使`A aa2 = 1;`合法,但`explicit`关键字可阻止这种转换。C++11起,成员变量可设默认值,如`int _b1 = 1;`。博客探讨构造函数、初始化列表及编译器优化,关注更多C++特性。
|
28天前
|
存储 编译器 C语言
【C++】类和对象②(类的默认成员函数:构造函数 | 析构函数)
C++类的六大默认成员函数包括构造函数、析构函数、拷贝构造、赋值运算符、取地址重载及const取址。构造函数用于对象初始化,无返回值,名称与类名相同,可重载。若未定义,编译器提供默认无参构造。析构函数负责对象销毁,名字前加`~`,无参数无返回,自动调用以释放资源。一个类只有一个析构函数。两者确保对象生命周期中正确初始化和清理。
|
2月前
|
C++ Linux
|
1月前
|
编译器 C语言 C++
【C++】:构造函数和析构函数
【C++】:构造函数和析构函数
26 0
|
2月前
|
编译器 C++
C++的基类和派生类构造函数
在 C++ 中,类的构造函数不能被继承,但基类的普通成员函数可以在派生类中访问。派生类必须通过其构造函数初始化继承的成员变量,由于私有成员变量无法直接初始化,因此需要在派生类构造函数中调用基类的构造函数来完成。示例代码显示了如何在派生类构造函数中调用基类构造函数,确保正确初始化。构造函数的调用顺序遵循自顶向下、从基类到派生类的规则,且只能调用直接基类的构造函数。如果基类没有默认构造函数,而派生类未指定构造函数调用,会导致编译错误。
36 4