8

我正在研究一个带有函数的 COM 对象库,该函数返回 a和VARIANTa SAFEARRAYBSTR如何显示此VARIANT实例中的值并将其保存在TStringList? 我试图在没有明确答案的情况下搜索网络。

我尝试了以下但没有成功:

Variant V;
String mystr;

VarClear(V);
TVarData(V).VType = varOleStr;
V = ComFunction->GetValues();  //<<<<----- V is empty
mystr = (wchar_t *)(TVarData(V).VString);
Memo1->Lines->Add(mystr);
VarClear(V);
4

3 回答 3

4
uses ActiveX;

var
  VSafeArray: PSafeArray;
  LBound, UBound, I: LongInt;
  W: WideString;
begin
  VSafeArray := ComFunction.GetValues();
  SafeArrayGetLBound(VSafeArray, 1, LBound);
  SafeArrayGetUBound(VSafeArray, 1, UBound);
  for I := LBound to UBound do
  begin
    SafeArrayGetElement(VSafeArray, I, W);
    Memo1.Lines.Add(W);
  end;
  SafeArrayDestroy(VSafeArray); // cleanup PSafeArray

如果您ComFunction通过后期绑定 ( CreateOleObject) 创建,您应该使用:

var
  v: Variant;
v := ComFunction.GetValues;
for i := VarArrayLowBound(v, 1) to VarArrayHighBound(v, 1) do
begin 
  W := VarArrayGet(v, [i]);
  Memo1.Lines.Add (W);
end;
于 2011-12-19T11:01:48.213 回答
4

您可以使用TWideStringDynArray并让 Delphi 进行转换:

procedure LoadStringsFromVariant(const Values: TWideStringDynArray; Strings: TStrings);
var
  I: Integer;
begin
  Strings.BeginUpdate;
  try
    for I := Low(Values) to High(Values) do
      Strings.Add(Values[I]);
  finally
    Strings.EndUpdate;
  end;
end;

当您使用 BSTR 的 Variant safearray 调用它时,它将TWideStringDynArray自动转换为。不兼容的 Variant 将导致运行时错误EVariantInvalidArgError

要检查 Variant 是否包含 BSTR 的安全数组,您可以执行以下操作:

IsOK := VarIsArray(V) and (VarArrayDimCount(V) = 1) and (VarType(V) and varTypeMask = varOleStr);
于 2011-12-19T08:27:12.347 回答
2

如何显示此 VARIANT 实例的值并将其保存在 TStringList 中?

COMVARIANT结构具有指向 a 的指针parraypparray数据成员SAFEARRAY,例如:

VARIANT V;
LPSAFEARRAY sa = V_ISBYREF(&V) ? V_ARRAYREF(&V) : V_ARRAY(&V);

Variant另一方面,VCL类LPSAFEARRAY定义了转换运算符,因此您可以直接分配它(但Variant.VType前提是不存在varByRef标志的字段),例如:

Variant V;
LPSAFEARRAY sa = V;

无论哪种方式,一旦你有了SAFEARRAY指针,就可以使用 SafeArray API 来访问这些BSTR值,例如:

bool __fastcall VariantToStrings(const Variant &V, TStrings *List)
{
    // make sure the Variant is holding an array
    if (!V_ISARRAY(&V)) return false;

    // get the array pointer
    LPSAFEARRAY sa = V_ISBYREF(&V) ? V_ARRAYREF(&V) : V_ARRAY(&V);

    // make sure the array is holding BSTR values
    VARTYPE vt;
    if (FAILED(SafeArrayGetVartype(sa, &vt))) return false;
    if (vt != VT_BSTR) return false;

    // make sure the array has only 1 dimension
    if (SafeArrayGetDim(sa) != 1) return false;

    // get the bounds of the array's sole dimension
    LONG lBound = -1, uBound = -1;
    if (FAILED(SafeArrayGetLBound(sa, 0, &lBound))) return false;
    if (FAILED(SafeArrayGetUBound(sa, 0, &uBound))) return false;

    if ((lBound > -1) && (uBound > -1))
    {
        // access the raw data of the array
        BSTR *values = NULL;
        if (FAILED(SafeArrayAccessData(sa, (void**)&values))) return false;
        try
        {
            List->BeginUpdate();
            try
            {
                // loop through the array adding the elements to the list
                for (LONG idx = lBound; l <= uBound; ++idx)
                {
                    String s;
                    if (values[idx] != NULL)
                        s = String(values[idx], SysStringLen(values[idx]));
                    List->Add(s);
                }
            }
            __finally
            {
                List->EndUpdate();
            }
        }
        __finally
        {
            // unaccess the raw data of the array
            SafeArrayUnaccessData(sa);
        }
    }

    return true;
}

变量清除(V);TVarData(V).VType = varOleStr;

你根本不需要那些。VCLVariant类将自身初始化为空白状态,并且无需分配 ,因为您随后会立即VType为整个分配一个新值。Variant

V = ComFunction->GetValues(); //<<<<----- V 为空

如果 V 为空,则GetValues()返回一个空Variant开始。

mystr = (wchar_t *)(TVarData(V).VString);

TVarData::VStringAnsiString&引用,而不是wchar_t*指针。要将 VCL Variant(不是 COM VARIANT)转换为 a String,只需按原样分配它,然后让 RTL 为您计算出细节:

String mystr = V;
于 2011-12-20T09:41:49.577 回答