2

v8sharp プロジェクトに機能を追加しようとしていますが、いくつかの問題があります (私は C++-CLI があまり得意ではないので、v8 を誤用するのではなく、C++-CLI の能力が不足していることに問題があると確信しています)。 .)

v8value.cpp:

v8sharp::V8FunctionWrapper^ V8ValueWrapper::WrapFunction(v8::Handle<v8::Value> value) {

  // Now we use the wrapper to make this safe to use
  // this works
  Console::WriteLine("IsFunction-First: {0}", value->IsFunction());
                // persistent so it doesn't get garbage collected
  v8::Persistent<v8::Value> pval(value);
                // create a function wrapper
  V8FunctionWrapper^ bla = gcnew V8FunctionWrapper(pval);
  return bla;
 }

which はv8Handle<v8::Value>関数を含む (この関数を呼び出すものによって常にそうなる) を取り、C# プロジェクトで使用できるように素敵な .net ラッパーを返す必要があります。

問題はここにあります v8functionwrapper.cpp:

#include "v8functionwrapper.h"
#include "v8value.h";


v8sharp::V8FunctionWrapper::V8FunctionWrapper(v8::Persistent<v8::Value> value)
{
    // is this wrong? 
 this->_value = &value;
     // still true
 Console::WriteLine("Constructor: {0}", (*this->_value)->IsFunction());


}

// This function is called by C# and triggers the javascript function
Object^ v8sharp::V8FunctionWrapper::Call([ParamArray]array<Object ^> ^ args)
{
 // Get a refence to the function
 Console::WriteLine("Cast 2");
    // MEMORY VIOLATION: the _value no longer points to a valid object :(
 Console::WriteLine("IsFunction: {0}", (*this->_value)->IsFunction());
 Console::ReadLine();
-- snip --

}

v8functionwrapper.h:

#pragma once
#include "v8.h"

using namespace System;
using namespace System::Reflection;

namespace v8sharp {
public ref class V8FunctionWrapper 
{
public:
 delegate bool V8FunctionCallback(array<Object ^> ^ args);
 Object^ v8sharp::V8FunctionWrapper::Call([ParamArray]array<Object ^> ^ args);
 v8::Persistent<v8::Value> Unwrap();
 V8FunctionWrapper(v8::Persistent<v8::Value> value);
 ~V8FunctionWrapper();
 !V8FunctionWrapper();

private:
 V8FunctionCallback^ _callback;
 v8::v8<Persistent::Value>* _value;

};
}

この行から明らかです (デバッグ コード): Console::WriteLine("IsFunction: {0}", (*this->_value)->IsFunction()); ポインター _value は無効になり、例外が発生します。ポインターが無効になるのはなぜですか? コンストラクターで引数を指していて、それが削除されるからですか? もしそうなら、どうすれば消えないポインターを取得できますか。これは .net クラスであるため、ネイティブ型を混在させることはできません。

4

1 に答える 1

1

新しい v8::Persistent 値をクラスのメンバーとしてインスタンス化する必要があります。これは、渡す値がスタック上に作成され、WrapFunction が戻るとすぐに破棄されるためです。また、オブジェクトが破棄されたときに _value を削除することを忘れないでください。

v8sharp::V8FunctionWrapper::V8FunctionWrapper(v8::Persistent<v8::Value> value)
{
    this->_value = new v8::Persistent<v8::Value>(value)
}
于 2010-06-30T01:29:30.793 に答える