3

所以,我想将 C++ 代码中的映射传递给 js 脚本。键是数字,值是包含使用普通 Node.js 技术包装的对象的列表。那是代码:

Handle<Value> topologicalSortedGraph( const Arguments &args ) {
    HandleScope scope;
    ...
    QMap<int, QList<Actor *> > topologicalSortedGraph = scriptContext->getTopologicalSortedGraph();
    const int schemeTiersCount = topologicalSortedGraph.size();
    Local<Object> scheme = Object::New( );
    for ( int i = 0; schemeTiersCount > i; ++i ) {
        Local<Object> tier = Object::New( );
        foreach ( Actor *actor, topologicalSortedGraph[i] ) {
            Handle<Value> actorInitData[] = { Int32::New( reinterpret_cast<int>( actor ) ) };
            Handle<Value> wrappedActor = ActorWrap::newInstance( 1, actorInitData );
            tier->Set( String::NewSymbol( actor->getId( ).toLocal8Bit( ).constData( ) ),
                wrappedActor );
        }
        scheme->Set( i, tier );
    }
    return scope.Close( scheme );

}

然后我在脚本中调用以下代码:

var addon = require('./Addon');
...
var scheme = addon.topologicalSortedGraph();
for (var number in scheme) {
    for (var actor in scheme[number]) {
        if (actor.isReady()) {
            addon.tick(actor);
            break;
        }
    }
}
...

问题是脚本无法识别“actor”对象,并在调用“actor.isReady()”时打印到控制台“undefined”。顺便说一句,这是“ActorWrap”类的定义:

class ActorWrap : public node::ObjectWrap {
public:
    static void                         init( );
    static Handle<Value>                newInstance( int argc, const Handle<Value> *argv );

private:
                                        ActorWrap( const Actor *initActor );
                                        ~ActorWrap( );

    static Handle<Value>                newObject( const Arguments &args );
    static Handle<Value>                id( const Arguments &args );
    static Handle<Value>                label( const Arguments &args );
    static Handle<Value>                isDone( const Arguments &args );
    static Handle<Value>                isReady( const Arguments &args );

    static Persistent<Function>         CONSTRUCTOR;
    static const char *                 CLASS_NAME;

    const Actor *                       actor;
};

和实施:

Persistent<Function> ActorWrap::CONSTRUCTOR;
const char *ActorWrap::CLASS_NAME = "Actor";

ActorWrap::ActorWrap( const Actor *initActor ) : actor( initActor ) {

}

ActorWrap::~ActorWrap( ) {

}

void ActorWrap::init( ) {
    Local<FunctionTemplate> tpl = FunctionTemplate::New( newObject );
    tpl->SetClassName( String::NewSymbol( CLASS_NAME ) );
    tpl->InstanceTemplate()->SetInternalFieldCount( 1 );

    tpl->PrototypeTemplate( )->Set( String::NewSymbol( "id" ),
        FunctionTemplate::New( id )->GetFunction( ) );
    tpl->PrototypeTemplate( )->Set( String::NewSymbol( "label" ),
        FunctionTemplate::New( label )->GetFunction( ) );
    tpl->PrototypeTemplate( )->Set( String::NewSymbol( "isDone" ),
        FunctionTemplate::New( isDone )->GetFunction( ) );
    tpl->PrototypeTemplate( )->Set( String::NewSymbol( "isReady" ),
        FunctionTemplate::New( isReady )->GetFunction( ) );

    CONSTRUCTOR = Persistent<Function>::New( tpl->GetFunction( ) );
}

Handle<Value> ActorWrap::newInstance( int argc, const Handle<Value> *argv ) {
    HandleScope scope;
    Handle<Value> objectInitData[] = { argv[0] };
    Local<Object> instance = CONSTRUCTOR->NewInstance( 1, objectInitData );
    return scope.Close( instance );
}

Handle<Value> ActorWrap::newObject( const Arguments &args ) {
    HandleScope scope;
    const Actor *actor = reinterpret_cast<Actor *>( args[0]->Int32Value( ) );
    Q_ASSERT( NULL != actor );
    ActorWrap *obj = new ActorWrap( actor );
    obj->Wrap( args.This( ) );
    return args.This( );
}

Handle<Value> ActorWrap::id( const Arguments &args ) {
    HandleScope scope;
    ActorWrap* obj = ObjectWrap::Unwrap<ActorWrap>( args.This( ) );
    return scope.Close( String::New( obj->actor->getId( ).toLocal8Bit( ).constData( ) ) );
}

Handle<Value> ActorWrap::label( const Arguments &args ) {
    HandleScope scope;
    ActorWrap* obj = ObjectWrap::Unwrap<ActorWrap>( args.This( ) );
    return scope.Close( String::New( obj->actor->getLabel( ).toLocal8Bit( ).constData() ) );
}

Handle<Value> ActorWrap::isDone( const Arguments &args ) {
    HandleScope scope;
    ActorWrap* obj = ObjectWrap::Unwrap<ActorWrap>( args.This( ) );
    LocalWorkflow::BaseWorker *worker = obj->actor->castPeer<BaseWorker>( );
    Q_ASSERT( NULL != worker );
    return scope.Close( Boolean::New( worker->isDone() ) );
}

Handle<Value> ActorWrap::isReady( const Arguments &args ) {
    HandleScope scope;
    ActorWrap* obj = ObjectWrap::Unwrap<ActorWrap>( args.This( ) );
    LocalWorkflow::BaseWorker *worker = obj->actor->castPeer<BaseWorker>( );
    Q_ASSERT( NULL != worker );
    return scope.Close( Boolean::New( worker->isReady() ) );
}

我怀疑在第一个片段中初始化“层”对象时出现问题。我没有找到任何关于将对象设置为另一个对象的属性的信息,所以当我只调用“obj1->Set(string, obj2);”时可能是我错了

4

1 回答 1

0

嗯,这个问题是微不足道的。由于我对 js 很陌生,我在迭代关联数组的代码中出现了错误。正确的变体如下

var addon = require('./Addon');
...
var scheme = addon.topologicalSortedGraph();
for (var number in scheme) {
    for (var actor in scheme[number]) {
        if (scheme[number][actor].isReady()) {
            addon.tick(actor);
            break;
        }
    }
}
...

所有 C++ 代码都是正确的,并且包装的对象也是正确的。

于 2013-03-25T08:31:12.353 回答