Microsoft MVP성태의 닷넷 이야기
.NET Framework: 449. C#에서 C++로 VARIANT 넘겨주는 방법 [링크 복사], [링크+제목 복사],
조회: 22926
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
[TestApp.zip]    

C#에서 C++로 VARIANT 넘겨주는 방법

그냥 방법 위주로 결과만 나열해 보겠습니다.

우선 C/C++ DLL에서 제공하는 함수가 다음과 같은 경우,

WIN32LIB_API int VariantFromCS(VARIANT var)
{
    if (var.vt != VT_BSTR)
    {
        return 0;
    }

    CComBSTR bstr = var.bstrVal;
    return bstr.Length();
}

C#에서 이 함수에 값을 제공하는 방법은 우선 VARIANT 구조체를 만듭니다.

// VARIANTARG (Structures)
// http://www.pinvoke.net/default.aspx/Structures/VARIANTARG.html

[StructLayout(LayoutKind.Explicit, Size = 16)]
struct VARIANTARG
{
    [FieldOffset(0)]
    public ushort vt;

    [FieldOffset(2)]
    public ushort wReserved1;

    [FieldOffset(4)]
    public ushort wReserved2;

    [FieldOffset(6)]
    public ushort wReserved3;

    [FieldOffset(8)]
    public long llVal;

    [FieldOffset(8)]
    public int lVal;

    [FieldOffset(8)]
    public byte bVal;

    [FieldOffset(8)]
    public short iVal;

    [FieldOffset(8)]
    public float fltVal;

    [FieldOffset(8)]
    public double dblVal;

    [FieldOffset(8)]
    public short boolVal;

    [FieldOffset(8)]
    public int scode;

    [FieldOffset(8)]
    public double date;

    [FieldOffset(8)]
    public unsafe ushort* bstrVal;

    [FieldOffset(8)]
    public unsafe byte* pbVal;

    [FieldOffset(8)]
    public unsafe short* piVal;

    [FieldOffset(8)]
    public unsafe int* plVal;

    [FieldOffset(8)]
    public unsafe long* pllVal;

    [FieldOffset(8)]
    public unsafe float* pfltVal;

    [FieldOffset(8)]
    public unsafe double* pdblVal;

    [FieldOffset(8)]
    public unsafe short* pboolVal;

    [FieldOffset(8)]
    public unsafe int* pscode;

    [FieldOffset(8)]
    public unsafe double* pdate;

    [FieldOffset(8)]
    public unsafe ushort** pbstrVal;

    [FieldOffset(8)]
    public unsafe VARIANTARG* pvarVal;

    [FieldOffset(8)]
    public unsafe void* byref;

    [FieldOffset(8)]
    public sbyte cVal;

    [FieldOffset(8)]
    public ushort uiVal;

    [FieldOffset(8)]
    public uint ulVal;

    [FieldOffset(8)]
    public ulong ullVal;

    [FieldOffset(8)]
    public int intVal;

    [FieldOffset(8)]
    public uint uintVal;

    [FieldOffset(8)]
    public unsafe sbyte* pcVal;

    [FieldOffset(8)]
    public unsafe ushort* puiVal;

    [FieldOffset(8)]
    public unsafe uint* pulVal;

    [FieldOffset(8)]
    public unsafe ulong* pullVal;

    [FieldOffset(8)]
    public unsafe int* pintVal;

    [FieldOffset(8)]
    public unsafe uint* puintVal;

    [FieldOffset(8)]
    public unsafe void* pvRecord;
}

이제 C#에서 문자열을 전달하고 싶다면 다음과 같이 코딩하면 됩니다.

[DllImport("Win32Lib.dll")]
static extern int VariantFromCS(VARIANTARG var);

static void Main(string[] args)
{
    VARIANTARG var = new VARIANTARG();
    var.vt = 8; // VT_BSTR == 8

    string txt = "test is good";
    fixed (char* str = txt)
    {
        var.bstrVal = (ushort*)str;
        Console.WriteLine("VariantFromCS Length: " + VariantFromCS(var) + " == " + txt.Length); 

        // 출력 결과
        // VariantFromCS Length: 12 == 12
    }
}




C/C++ 측에서 "VARIANT *"를 받는 요구한다면 어떻게 해야 할까요?

WIN32LIB_API int PointerVariantFromCS(VARIANT *pVar)
{
    if (pVar->vt != VT_BSTR)
    {
        return 0;
    }

    CComBSTR bstr = pVar->bstrVal;
    return bstr.Length();
}

상관없습니다. ^^ C# 측에서 맞춰주기만 하면 됩니다.

[DllImport("Win32Lib.dll")]
static unsafe extern int PointerVariantFromCS(VARIANTARG* pVar);

static void Main(string[] args)
{
    unsafe
    {
        VARIANTARG var = new VARIANTARG();
        var.vt = 8; // VT_BSTR == 8

        string txt = "test is good";

        fixed (char* str = txt)
        {
            var.bstrVal = (ushort*)str;
            Console.WriteLine("PointerVariantFromCS Length: " + PointerVariantFromCS(&var) + " == " + txt.Length);
        
            // 출력 결과
            // PointerVariantFromCS Length: 12 == 12
        }
    }
}




VARIANT의 배열을 받는 경우라면 어떨까요?

WIN32LIB_API int ArrayVariantFromCS(VARIANT *pVar)
{
    int totalLength = 0;

    for (VARIANT *first = pVar; (*first).vt != VT_EMPTY; first++)
    {
        CComBSTR bstr = pVar->bstrVal;
        totalLength += bstr.Length();
    }

    return totalLength;
}

역시 상관없습니다. C/C++에서처럼 하듯이 C#에서도 직관적으로 맞춰주면 됩니다.

[DllImport("Win32Lib.dll")]
static unsafe extern int ArrayVariantFromCS(VARIANTARG[] pVar);

static void Main(string[] args)
{
    unsafe
    {
        string txt = "test is good";
        string txt2 = "this is test";

        fixed (char* str1 = txt)
        fixed (char* str2 = txt2)
        {
            VARIANTARG[] varArray = new VARIANTARG[3];
            varArray[0] = new VARIANTARG();
            varArray[1] = new VARIANTARG();
            varArray[2] = new VARIANTARG();

            varArray[0].vt = 8;
            varArray[0].bstrVal = (ushort*)str1;

            varArray[1].vt = 8;
            varArray[1].bstrVal = (ushort*)str2;

            varArray[2].vt = 0; // VT_EMPTY == 0

            Console.WriteLine("ArrayVariantFromCS Length: " + ArrayVariantFromCS(varArray) + " == " + (txt.Length + txt2.Length));

            // 출력 결과
            // ArrayVariantFromCS Length: 24 == 24
        }
    }
}

예전에 비슷한 주제로 한번 이야기 한 적이 있지요. ^^

How to Interop DISPPARAMS
; https://www.sysnet.pe.kr/2/0/617

위의 방법을 써서 ArrayVariantFromCS 함수에 전달하는 것을 다음과 같이 우회하는 것도 가능합니다.

Guid[] guids = new Guid[2];
IntPtr pGuid = IntPtr.Zero;
fixed (void* pArrayGuid0 = &guids[0])
{
    pGuid = new IntPtr(pArrayGuid0);
    Marshal.GetNativeVariantForObject(txt, pGuid);
}

fixed (void* pArrayGuid1 = &guids[1])
{
    pGuid = new IntPtr(pArrayGuid1);
    Marshal.GetNativeVariantForObject(txt2, pGuid);
}

GCHandle gcHandle = GCHandle.Alloc(guids, GCHandleType.Pinned);
{
    IntPtr argPtr = gcHandle.AddrOfPinnedObject();
    Console.WriteLine("ArrayVariantFromCS(2) Length: " + ArrayVariantFromCS(argPtr));

    // 출력 결과
    // ArrayVariantFromCS(2) Length: 24

    gcHandle.Free();
}

C/C++ 언어와 이렇게나 호환이 잘 되니... 정말이지 C#이란 언어는 사랑하지 않을래야 않을 수가 없습니다. ^^

뭐 이정도면... ^^ 언제든 쉽게 가져다 쓰실 수 있겠죠!

(첨부 파일은 위의 예제 코드를 모두 포함하는 C#과 C++프로젝트입니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 7/5/2014]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 151  152  153  154  [155]  156  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1177정성태11/18/201129924.NET Framework: 272. 소켓 연결 시간 제한 - 두 번째 이야기 [1]파일 다운로드1
1176정성태11/17/201129147.NET Framework: 271. C#에서 확인해 보는 관리 힙의 인스턴스 구조 [3]파일 다운로드1
1175정성태11/16/201127136.NET Framework: 270. .NET 참조 개체 인스턴스의 Object Header를 확인하는 방법 [1]파일 다운로드1
1174정성태11/15/201126553.NET Framework: 269. 일반 참조형의 기본 메모리 소비는 얼마나 될까요? [4]
1173정성태11/14/201122761.NET Framework: 268. .NET Array는 왜 12bytes의 기본 메모리를 점유할까? [1]
1172정성태11/13/201119727.NET Framework: 267. windbg - GC Heap에서 .NET 타입에 대한 배열을 찾는 방법
1171정성태11/12/201136413.NET Framework: 266. StringBuilder에서의 OutOfMemoryException 오류 원인 분석 [4]파일 다운로드1
1170정성태11/10/201125622.NET Framework: 265. Named 동기화 개체 생성 시 System.UnauthorizedAccessException 예외 발생하는 경우
1169정성태11/10/201129425.NET Framework: 264. 다중 LAN 카드 환경에서 Dns.GetHostAddresses(local)가 반환해 주는 IP의 우선순위는 어떻게 될까요? [4]
1168정성태11/6/201125284오류 유형: 139. TlbImp : error TI0000 : A single valid machine type compatible with the input type library must be specified
1167정성태11/5/201137116개발 환경 구성: 133. Registry 등록 과정 없이 COM 개체 사용 - 두 번째 이야기 [5]파일 다운로드4
1166정성태11/5/201123143.NET Framework: 263. byte[] pData = new byte[100000]로 인한 성능 차이? [1]파일 다운로드1
1165정성태11/3/201128051개발 환경 구성: 132. "Visual Studio Command Prompt (2010)" 명령행에서 2.0 버전의 MSBuild를 구동하는 방법 [2]파일 다운로드1
1164정성태11/1/201126219.NET Framework: 262. .NET 스레드 콜 스택 덤프 (4) - .NET 4.0을 지원하지 않는 MSE 응용 프로그램 원인 분석
1163정성태10/31/201125726.NET Framework: 261. .NET 스레드 콜 스택 덤프 (3) - MSE 소스 코드 개선파일 다운로드1
1162정성태10/30/201125850.NET Framework: 260. .NET 스레드 콜 스택 덤프 (2) - Managed Stack Explorer 소스 코드를 이용한 스택 덤프 구하는 방법파일 다운로드1
1161정성태10/29/201122640.NET Framework: 259. Type.GetMethod - System.Reflection.AmbiguousMatchException파일 다운로드1
1159정성태10/28/201126094.NET Framework: 258. Roslyn 맛보기 - SyntaxTree 조작 [2]
1158정성태10/24/201125411.NET Framework: 257. Roslyn 맛보기 - Roslyn Symbol / Binding API파일 다운로드1
1157정성태10/23/201129865.NET Framework: 256. Roslyn 맛보기 - Syntax Analysis (Roslyn Syntax API) [2]
1156정성태10/23/201128315.NET Framework: 255. Roslyn 맛보기 - Roslyn Services APIs를 이용한 Code Issue 및 Code Action 기능 소개 [1]
1155정성태10/22/201126381.NET Framework: 254. Roslyn 맛보기 - C# Interactive (2)
1154정성태10/22/201133120.NET Framework: 253. Roslyn 맛보기 - C# Interactive (1)
1153정성태10/21/201141990.NET Framework: 252. Roslyn 맛보기 - C# 소스 코드를 스크립트처럼 다루는 방법 [7]파일 다운로드1
1152정성태10/20/201123677.NET Framework: 251. string.GetHashCode는 hash 값을 cache 할까?
1151정성태10/18/201122598Java: 13. 자바도 64비트에서 (2GB) OutOfMemoryException 예외가 발생할까?
... 151  152  153  154  [155]  156  157  158  159  160  161  162  163  164  165  ...