使用Node.js Addon实现类继承

本文转载自微信公众号「编程杂技」,作者theanarkh。转载本文请联系编程杂技公众号。

我们提供的服务有:网站制作、成都网站建设、微信公众号开发、网站优化、网站认证、北屯ssl等。为1000+企事业单位解决了网站和推广的问题。提供周到的售前咨询和贴心的售后服务,是有科学管理、有技术的北屯网站制作公司

前言:昨天有个同学问怎么通过NAPI把C++类的继承关系映射到JS,很遗憾,NAPI貌似还不支持,但是V8支持,因为V8在头文件里导出了这些API,并Node.js里也依赖这些API,所以可以说是比较稳定的。本文介绍一下如何实现这种映射(不确定是否能满足这位同学的需求)。

下面我们看一下Addon的实现。会涉及到V8的一些使用,可以先阅读该文章《一段js理解nodejs中js调用c++/c的过程》。首先看一下基类的实现。

 
 
 
  1. #ifndef BASE_H 
  2. #define BASE_H 
  3. #include  
  4. #include  
  5. #include  
  6.  
  7. using namespace node; 
  8. using namespace v8; 
  9. class Base: public ObjectWrap { 
  10.     public: 
  11.         static void New(const FunctionCallbackInfo& info) { 
  12.             // 新建一个对象,然后包裹到info.This()中,后面会解包出来使用 
  13.             Base* base =  new Base(); 
  14.             base->Wrap(info.This()); 
  15.         } 
  16.  
  17.         static void Print(const FunctionCallbackInfo& info) { 
  18.             // 解包出来使用 
  19.             Base* base = ObjectWrap::Unwrap(info.This()); 
  20.             base->print(); 
  21.         } 
  22.  
  23.         void print() { 
  24.             printf("base print\n"); 
  25.         } 
  26.  
  27.         void hello() { 
  28.             printf("base hello\n"); 
  29.         } 
  30. }; 
  31.  
  32. #endif 

Node.js提供的ObjectWrap类实现了Wrap和UnWrap的功能,所以我们可以继承它简化封包解包的逻辑。Base类定义了两个功能函数hello和print,同时定义了两个类静态函数New和Print。New函数是核心逻辑,该函数在js层执行new Base的时候会执行并传入一个对象,这时候我们首先创建一个真正的有用的对象,并且通过Wrap把该对象包裹到传进来的对象里。我们继续看一下子类。

 
 
 
  1. #ifndef DERIVED_H 
  2. #define DERIVED_H 
  3. #include  
  4. #include  
  5. #include"Base.h" 
  6.  
  7. using namespace node; 
  8. using namespace v8; 
  9. class Derived: public Base { 
  10.     public: 
  11.         static void New(const FunctionCallbackInfo& info) { 
  12.             Derived* derived =  new Derived(); 
  13.             derived->Wrap(info.This()); 
  14.         } 
  15.  
  16.         static void Hello(const FunctionCallbackInfo& info) { 
  17.             Derived* derived = ObjectWrap::Unwrap(info.This()); 
  18.             // 调用基类的函数 
  19.             derived->hello(); 
  20.         } 
  21. }; 
  22.  
  23. #endif 

子类的逻辑类似,New函数和基类的逻辑一样,除了继承基类的方法外,额外定义了一个Hello函数,但是我们看到这只是个壳子,底层还是调用了基类的函数。定义完基类和子类后,我们把这两个类导出到JS。

 
 
 
  1. #include  
  2. #include "Base.h" 
  3. #include "Derived.h" 
  4.  
  5. namespace demo { 
  6.  
  7. using v8::FunctionCallbackInfo; 
  8. using v8::Isolate; 
  9. using v8::Local; 
  10. using v8::Object; 
  11. using v8::String; 
  12. using v8::Value; 
  13. using v8::FunctionTemplate; 
  14. using v8::Function; 
  15. using v8::Number; 
  16. using v8::MaybeLocal; 
  17. using v8::Context; 
  18. using v8::Int32; 
  19. using v8::NewStringType; 
  20.  
  21. void Initialize( 
  22.   Local exports, 
  23.   Local module, 
  24.   Local context 
  25. ) { 
  26.   Isolate * isolate = context->GetIsolate(); 
  27.   // 新建两个函数模板,基类和子类,js层New导出的函数时,V8会执行New函数并传入一个对象 
  28.   Local base = FunctionTemplate::New(isolate, Base::New); 
  29.   Local derived = FunctionTemplate::New(isolate, Derived::New); 
  30.  
  31.   // js层使用的类名 
  32.   NewStringType type = NewStringType::kNormal; 
  33.   Local base_string = String::NewFromUtf8(isolate, "Base", type).ToLocalChecked(); 
  34.   Local derived_string = String::NewFromUtf8(isolate, "Derived", type).ToLocalChecked(); 
  35.  
  36.   // 预留一个指针空间 
  37.   base->InstanceTemplate()->SetInternalFieldCount(1); 
  38.   derived->InstanceTemplate()->SetInternalFieldCount(1); 
  39.  
  40.   // 定义两个函数模板,用于属性的值 
  41.   Local BasePrint = FunctionTemplate::New(isolate, Base::Print); 
  42.   Local Hello = FunctionTemplate::New(isolate, Derived::Hello); 
  43.  
  44.   // 给基类定义一个print函数 
  45.   base->PrototypeTemplate()->Set(isolate, "print", BasePrint); 
  46.   // 给子类定义一个hello函数 
  47.   derived->PrototypeTemplate()->Set(isolate, "hello", Hello); 
  48.   // 建立继承关系 
  49.   derived->Inherit(base); 
  50.   // 导出两个函数给js层 
  51.   exports->Set(context, base_string, base->GetFunction(context).ToLocalChecked()).Check(); 
  52.   exports->Set(context, derived_string, derived->GetFunction(context).ToLocalChecked()).Check(); 
  53.  
  54. NODE_MODULE_CONTEXT_AWARE(NODE_GYP_MODULE_NAME, Initialize) 
  55. 我们看到给基类原型定义了一个print函数,给子类定义了hello函数。最后我们看看如何在JS层使用。

     
     
     
    1. const { Base, Derived } = require('./build/Release/test.node'); 
    2. const base = new Base(); 
    3. const derived = new Derived(); 
    4. base.print(); 
    5. derived.hello(); 
    6. derived.print(); 
    7. console.log(derived instanceof Base, derived instanceof Derived) 

    下面是具体的输出

     
     
     
    1. base print 
    2. base hello 
    3. base print 
    4. true true 

    我们逐句分析

    1 base.print()比较简单,就是调用基类定义的Print函数。

    2 derived.hello()看起来是调用了子类的Hello函数,但是Hello函数里调用了基类的hello函数,实现了逻辑的复用。

    3 derived.print()子类没有实现print函数,这里调用的是基类的print函数,和1一样。

    4 derived instanceof Base, derived instanceof Derived。根据我们的定义,derived不仅是Derived的实例,也是Base的实例。

    实现代码分析完了,我们看到把C++类映射到JS的方式有两种,第一种就是两个C++ 类没有继承关系,通过V8的继承API实现两个JS层存在继承关系的类(函数),比如print函数的实现,我们看到子类没有实现print,但是可以调用print,因为基类定义了,Node.js就是这样处理的。第二种就是两个存在继承关系的C++类,同样先通过V8的API实现两个继承的类导出到JS使用,因为JS层使用的只是壳子,具体执行到C++代码的时候,我们再体现出这种继承关系。比如Hello函数的实现,虽然我们是在子类里导出了hello函数,并且JS执行hello的时候的确执行到了子类的C++代码,但是最后会调用基类的hello函数。

    最后我们通过Nodej.js看看是如何做这种映射的,我们通过PipeWrap.cc的实现进行分析。

     
     
     
    1. // 新建一个函数模板 
    2. Local t = env->NewFunctionTemplate(New);// 继承两个函数模板 
    3. t->Inherit(LibuvStreamWrap::GetConstructorTemplate(env));// 导出给JS使用 
    4. exports->Set(env->context(), 
    5.               pipeString, 
    6.               t->GetFunction(env->context()).ToLocalChecked()).Check(); 

    上面代码实现了继承,我们看看GetConstructorTemplate的实现。

     
     
     
    1. tmpl = env->NewFunctionTemplate(nullptr); 
    2. env->SetProtoMethod(tmpl, "setBlocking", SetBlocking); 
    3. env->SetProtoMethod(tmpl, "readStart", JSMethod<&StreamBase::ReadStartJS>); 
    4. env->SetProtoMethod(t, "readStop", JSMethod<&StreamBase::ReadStopJS>);// ... 

    上面代码新建了一个新的函数模板并且设置了一系列的原型属性,那么模板t就继承了这些属性。我们看看Node.js里怎么使用的。

     
     
     
    1. function createHandle(fd, is_server) { 
    2.   // ... 
    3.   return new Pipe( 
    4.       is_server ? PipeConstants.SERVER : PipeConstants.SOCKET 
    5.   );} 
    6.  
    7. this._handle = createHandle(fd, false); 
    8. err = this._handle.setBlocking(true); 

    上面的代码首先会创建一个Pipe对象,然后调用它的setBlocking方法。我们发现Pipe(pipe_wrap.cc)是没有实现setBlocking函数的,但是好为什么他可以调用setBlocking呢?答案就是它的基类实现了。

    后记:在JS里实现继承是简单的,但是在底层实现起来还是比较复杂的,但是从代码设计的角度来看是非常有必要的。

    代码可以在仓库获取:

    https://github.com/theanarkh/learn-to-write-nodejs-addons。

    标题名称:使用Node.js Addon实现类继承
    URL链接:http://www.csdahua.cn/qtweb/news7/293257.html

    网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

    广告

    声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网