3

C++ アンマネージ コードから C# メソッドを呼び出しています。配列で返されたクラス インスタンスから値を取得する際に問題があります。

コードを少し簡略化しました

これが問題の方法です。

    [return: MarshalAs(UnmanagedType.SafeArray, SafeArraySubType = VarEnum.VT_UNKNOWN)]
    public ScOrder[] GetOrders()
    {
        return new ScOrder[] {

            (new ScOrder(1),
            (new ScOrder(2)
        };
    }

これは IScOrder インターフェイスです

[ComVisible(true)]
[Guid("B2B134CC-70A6-43CD-9E1E-B3A3D9992C3E")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IScOrder
{
    long GetQuantity();
}

これが ScOrder の実装です

[ComVisible(true)]
[Guid("F739759E-4D00-440E-B0B7-69AAF97FCB6D")]
[ClassInterface(ClassInterfaceType.None)]
public class ScOrder
{
    private long quantity = 0;

    public ScOrder() {}

    public ScOrder(long quantity)
    {
        this.quantity = quantity;
    }

    public long GetQuantity()
    {
        return this.quantity;
    }
}

これは、前回のリクエストで Zdeslav Vojkovic の助けを借りて作成した C++ コードです。問題はコメントに記載されています

  • 私はATLもMFCも使用していません。
  • C++ tlb ファイルは regasm によって生成されます。

COM の初期化と GetOrders メソッドの呼び出しはうまく機能します

IScProxyPtr iPtr;
CoInitialize(NULL);
iPtr.CreateInstance(CLSID_ScProxy);
SAFEARRAY* orders;
iPtr->GetOrders(&orders);
LPUNKNOWN* punks;
HRESULT hr = SafeArrayAccessData(orders, (void**)&punks);
if(SUCCEEDED(hr)) 
{
    long lbound, ubound;
    SafeArrayGetLBound(orders, 1, &lbound);
    SafeArrayGetUBound(orders, 1, &ubound);
    long elements = ubound - lbound + 1;
    for(int i=0;i<elements;i++) 
    {
        LPUNKNOWN punk = punks[i]; //the punk seems valid 
        IScOrderPtr order(punk); //unfortunatelly, "order" now points to {0x00000000}

        //subsequent attempt to get the value will fail
        long quantity = 0;
        HRESULT procCall;
        //GetQuantity will throw an exception
        procCall = order->GetQuantity((long long *)q); 

    }
    SafeArrayUnaccessData(orders);
}
SafeArrayDestroy(orders);

Zdeslav のおかげで、注文 (パンク) 内でデバッグできることがわかりました。

IScOrderPtr order(punk);

それで、そこで何が起こっているのかを見るために、オーダー(パンク)に足を踏み入れました。「comip.h」に入りました

// Constructs a smart-pointer from any IUnknown-based interface pointer.
//
template<typename _InterfaceType> _com_ptr_t(_InterfaceType* p) 
    : m_pInterface(NULL)
{
    HRESULT hr = _QueryInterface(p);

...次に、_QueryInterface(p) 実装の内部に足を踏み入れました。comip.h にもあります

// Performs a QI on pUnknown for the interface type returned
// for this class.  The interface is stored.  If pUnknown is
// NULL, or the QI fails, E_NOINTERFACE is returned and
// _pInterface is set to NULL.
//
template<typename _InterfacePtr> HRESULT _QueryInterface(_InterfacePtr p) throw()
{
    HRESULT hr;

    // Can't QI NULL
    //
    if (p != NULL) {
        // Query for this interface
        //
        Interface* pInterface;
        hr = p->QueryInterface(GetIID(), reinterpret_cast<void**>(&pInterface));

ここでの問題は、返される "hr" の値が E_NOINTERFACE であることです ... これは正しくありません。

私は C++ や COM の専門家ではありません...助けてください :)

4

1 に答える 1

1

あなたのクラスは、C# 側でインターフェイスをScOrder実装していないようです。IScOrder

次のようになっているはずです。

//[ComVisible(true)]
//[Guid("F739759E-4D00-440E-B0B7-69AAF97FCB6D")]
//[ClassInterface(ClassInterfaceType.None)]
public class ScOrder : IScOrder

上記でコメントし[...]たのは、干渉しているからではなく、必要がないように見えるからですIScOrder。COM の可視性が必要であり、C++ 側で取得できる必要があります。

IScOrderインスタンスを継承しないと、いくつかのインターフェースがありますが、関心のあるものIScOrderは実際にはポインターでアクセスできません。

于 2012-09-15T11:07:52.813 に答える