0

我已经编写了我的第一个Node.JS N-Api 插件,但它会因日志而崩溃:

internal/modules/cjs/loader.js:718
  return process.dlopen(module, path.toNamespacedPath(filename));
                 ^

Error: /home/d/Projects/engine/build/Release/engine.node: undefined symbol: _ZTV6Config
    at Object.Module._extensions..node (internal/modules/cjs/loader.js:718:18)
    at Module.load (internal/modules/cjs/loader.js:599:32)
    at tryModuleLoad (internal/modules/cjs/loader.js:538:12)
    at Function.Module._load (internal/modules/cjs/loader.js:530:3)
    at Module.require (internal/modules/cjs/loader.js:637:17)
    at require (internal/modules/cjs/helpers.js:22:18)
    at Object.<anonymous> (/home/d/Projects/engine/index.js:1:78)
    at Module._compile (internal/modules/cjs/loader.js:689:30)
    at Object.Module._extensions..js (internal/modules/cjs/loader.js:700:10)
    at Module.load (internal/modules/cjs/loader.js:599:32)

下面是我的代码:
wrapper.h

#include <napi.h>
#include "types.hpp"

using namespace std;

class Config: public Napi::ObjectWrap<Config> {
    config_t _cfg;
public:
    explicit Config(const Napi::CallbackInfo &info);
    ~Config();
    static Napi::FunctionReference constructor;
    static Napi::Object Init(Napi::Env &env, Napi::Object &exports);

    config_t Get();
private:
    Napi::Value GetTimeAccuracy(const Napi::CallbackInfo &info);
    void SetTimeAccuracy(const Napi::CallbackInfo &info, const Napi::Value &value);
};

包装器.cc

Config::Config(const Napi::CallbackInfo &info): Napi::ObjectWrap<Config>(info) {

}

Napi::Object Config::Init(Napi::Env &env, Napi::Object &exports) {
    // This method is used to hook the accessor and method callbacks
    Napi::Function func = DefineClass(env, "Config", {
        InstanceAccessor("timeAccuracy", &Config::GetTimeAccuracy, &Config::SetTimeAccuracy)
    });
    // Create a peristent reference to the class constructor. This will allow
    // a function called on a class prototype and a function
    // called on instance of a class to be distinguished from each other.
    constructor = Napi::Persistent(func);
    // Call the SuppressDestruct() method on the static data prevent the calling
    // to this destructor to reset the reference when the environment is no longer
    // available.
    constructor.SuppressDestruct();
    exports.Set("Config", func);
    return exports;
}

Napi::FunctionReference Config::constructor;

config_t Config::Get() {
    return _cfg;
}

Napi::Value Config::GetTimeAccuracy(const Napi::CallbackInfo &info) {
    Napi::Env env = info.Env();
    return Napi::Number::New(env, _cfg.time_accuracy);
}

void Config::SetTimeAccuracy(const Napi::CallbackInfo &info, const Napi::Value &value) {
    _cfg.time_accuracy = value.As<Napi::Number>().FloatValue();
}

引擎.cc

#include <napi.h>
#include "wrapper.h"

Napi::Object init(Napi::Env env, Napi::Object exports) {
    Config::Init(env, exports);
    return exports;
}

NODE_API_MODULE(engine, init);

绑定.gyp

{
    "targets": [{
        "cflags_cc": ["-std=c++17"],
        "include_dirs": [
            "<!@(node -p \"require('node-addon-api').include\")"
        ],
        "target_name": "engine",
        "sources": [
            "wrapper.cc",
            "engine.cc",
            "types.cpp",
        ],
        "defines": ["NAPI_DISABLE_CPP_EXCEPTIONS"]
    }]
}

当我重建项目时,它没有任何错误。之前我写了一个简单的Hello, World!我刚刚注册了打印文本的功能并且一切正常。但是现在当我尝试将我的模块导入为const engine = require('./build/Release/engine.node');.
当我得到我的模块寄存器但在 Node.JS 之后找不到Config.

我该如何解决?

4

1 回答 1

1

_ZTV6Config是. _ _vtable for Config

令人困惑的是,这实际上是由于Config.

编译器在找到析构函数定义时在同一个翻译单元中定义 vtable。

于 2020-02-13T18:27:32.487 回答