Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 2개 있습니다.)
.NET Framework: 446. Assembly.Load를 이용해 GAC에 등록된 어셈블리를 로드하는 방법
; https://www.sysnet.pe.kr/2/0/1703

.NET Framework: 579. Assembly.LoadFrom으로 로드된 어셈블리의 JIT 컴파일 코드 공유?
; https://www.sysnet.pe.kr/2/0/10953




Assembly.Load를 이용해 GAC에 등록된 어셈블리를 로드하는 방법

GAC에 등록된 어셈블리를 로드하는 방법이 애매합니다. 물론 다음과 같이 하면 어셈블리 로드가 됩니다.

string assemblyName = "Microsoft.SqlServer.Types, Version=12.0.0.0, Culture=neutral, PublicKeyToken=89845dcd8080cc91";
Assembly asm = Assembly.Load(assemblyName);

하지만 저렇게 어셈블리 이름을 완전하게 지정하는 것이 애매할 때가 있습니다. 가장 큰 문제는 Version 값인데요. 대상 PC에 어떤 버전의 어셈블리가 설치되어 있는지 확실하지 않을 때가 있습니다.

이런 경우 해결책은 2가지 정도가 있습니다.

  • Assembly.LoadWithPartialName을 이용하는 방법
  • 그냥 특정 버전의 어셈블리를 로컬에 복사해서 배포하는 방법

첫 번째 LoadWithPartialName은 현재 [obsolete] 특성이 적용된 상태라서 왠지 쓰기가 불안합니다. 두 번째 방법이 그나마 나은데요. 이것도 좀 그런 것이... 대상 PC의 GAC에 설치된 최신 버전의 어셈블리가 그 제품의 최신 버전과만 연동할 수 있는 경우가 가능하기 때문입니다.

방법이 없을까??? 하고 검색을 좀 해보았는데, 색다른 해결책이 하나 나왔습니다.

Is it possible to Load an assembly from the GAC without the FullName?
; http://stackoverflow.com/questions/6121276/is-it-possible-to-load-an-assembly-from-the-gac-without-the-fullname

fusion.dll의 어셈블리 정보를 반환하는 IAssemblyCache.QueryAssemblyInfo 메서드를 직접 사용하는 것입니다. 위의 코드를 대략 정리해 보면 다음과 같습니다.

using System;
using System.Reflection;
using System.Runtime.InteropServices;

namespace ConsoleApplication1
{
    class Program
    {
        [DllImport("fusion.dll")]
        private static extern int CreateAssemblyCache(out IAssemblyCache ppAsmCache, int reserved);

        static void Main(string[] args)
        {
            string gacAsmPath = GetAssemblyPath("Microsoft.SqlServer.Types");
            Assembly sqlServerTypes = Assembly.LoadFrom(gacAsmPath);
        }

        public static string GetAssemblyPath(string name)
        {
            if (name == null)
            {
                throw new ArgumentNullException("name");
            }

            string finalName = name;
            AssemblyInfo aInfo = new AssemblyInfo();
            aInfo.cchBuf = 1024;
            aInfo.currentAssemblyPath = new String('\0', aInfo.cchBuf);

            IAssemblyCache ac;
            int hr = CreateAssemblyCache(out ac, 0);
            if (hr >= 0)
            {
                hr = ac.QueryAssemblyInfo(0, finalName, ref aInfo);
                if (hr < 0)
                {
                    return null;
                }
            }

            return aInfo.currentAssemblyPath;
        }
    }

    [StructLayout(LayoutKind.Sequential)]
    public struct AssemblyInfo
    {
        public int cbAssemblyInfo;
        public int assemblyFlags;
        public long assemblySizeInKB;
        [MarshalAs(UnmanagedType.LPWStr)]
        public string currentAssemblyPath;
        public int cchBuf;
    }

    [ComImport, InterfaceType(ComInterfaceType.InterfaceIsIUnknown), Guid("e707dcde-d1cd-11d2-bab9-00c04f8eceae")]
    public interface IAssemblyCache
    {
        void Reserved0();

        [PreserveSig]
        int QueryAssemblyInfo(int flags, [MarshalAs(UnmanagedType.LPWStr)] string assemblyName, ref AssemblyInfo assemblyInfo);
    }
}

그런데, 아쉽게도 GetAssemblyPath 메서드는 해당 어셈블리의 GAC 전체 경로를 반환하는 것이지, 어셈블리 이름을 반환하지는 않습니다. 그래서 Assembly.LoadFrom을 사용해야 하는데 이것도 역시 문제가 될 수 있습니다. 왜냐하면, Load와 LoadFrom의 바인딩 문맥이 다르기 때문입니다.

BindingContext - Load() vs. LoadFrom()
; https://imhumanvirus.tistory.com/186

약간은 불편하지만, AppDomain을 이용해 안전하게 어셈블리 이름만 구할 수도 있습니다. 즉, 다음과 같이 구현하시면 끝!

static void Main(string[] args)
{
    string gacAsmPath = GetAssemblyPath("Microsoft.SqlServer.Types");

    AppDomain appDomain = AppDomain.CreateDomain(Guid.NewGuid().ToString());
    appDomain.SetData("asmName", gacAsmPath);
    appDomain.DoCallBack(
    () => {
        string asmName = AppDomain.CurrentDomain.GetData("asmName") as string;

        if (string.IsNullOrEmpty(asmName) == true)
        {
            return;
        }

        Assembly target = Assembly.LoadFrom(asmName);
        AppDomain.CurrentDomain.SetData("asmName", target.FullName);
    });

    string assemblyName = appDomain.GetData("asmName") as string;
    AppDomain.Unload(appDomain);

    Assembly asm = Assembly.Load(assemblyName);

    Console.WriteLine(assemblyName + " in " + asm.FullName);
}

(첨부 파일은 위의 예제를 포함하고 있습니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/21/2024]

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

비밀번호

댓글 작성자
 



2015-10-19 04시57분
.NET 어셈블리 로드 컨텍스트(최종이길 바라며...)
; https://imhumanvirus.tistory.com/315
정성태

... 61  62  63  64  65  66  67  68  69  [70]  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12228정성태6/12/202020716.NET Framework: 909. C# - Source Generator를 적용한 XmlCodeGenerator파일 다운로드1
12227정성태6/12/202024828오류 유형: 616. Visual Studio의 느린 업데이트 속도에 대한 원인 분석 [5]
12226정성태6/11/202023859개발 환경 구성: 493. OpenVPN의 네트워크 구성 [4]파일 다운로드1
12225정성태6/11/202021399개발 환경 구성: 492. 윈도우에 OpenVPN 설치 - 클라이언트 측 구성
12224정성태6/11/202030490개발 환경 구성: 491. 윈도우에 OpenVPN 설치 - 서버 측 구성 [1]
12223정성태6/9/202027303.NET Framework: 908. C# - Source Generator 소개 [10]파일 다운로드2
12222정성태6/3/202019225VS.NET IDE: 146. error information: "CryptQueryObject" (-2147024893/0x80070003)
12221정성태6/3/202019029Windows: 170. 비어 있지 않은 디렉터리로 symbolic link(junction) 연결하는 방법
12220정성태6/3/202023368.NET Framework: 907. C# DLL로부터 TLB 및 C/C++ 헤더 파일(TLH)을 생성하는 방법
12219정성태6/1/202021740.NET Framework: 906. C# - lock (this), lock (typeof(...))를 사용하면 안 되는 이유파일 다운로드1
12218정성태5/27/202020413.NET Framework: 905. C# - DirectX 게임 클라이언트 실행 중 키보드 입력을 감지하는 방법 [3]
12217정성태5/24/202018834오류 유형: 615. Transaction count after EXECUTE indicates a mismatching number of BEGIN and COMMIT statements. Previous count = 0, current count = 1.
12216정성태5/15/202022307.NET Framework: 904. USB/IP PROJECT를 이용해 C#으로 USB Keyboard 가상 장치 만들기 [14]파일 다운로드1
12215정성태5/12/202028222개발 환경 구성: 490. C# - (Wireshark의) USBPcap을 이용한 USB 패킷 모니터링 [10]파일 다운로드1
12214정성태5/5/202020574개발 환경 구성: 489. 정식 인증서가 있는 경우 Device Driver 서명하는 방법 (2) - UEFI/SecureBoot [1]
12213정성태5/3/202021033개발 환경 구성: 488. (User-mode 코드로 가상 USB 장치를 만들 수 있는) USB/IP PROJECT 소개
12212정성태5/1/202018234개발 환경 구성: 487. UEFI / Secure Boot 상태인지 확인하는 방법
12211정성태4/27/202021084개발 환경 구성: 486. WSL에서 Makefile로 공개된 리눅스 환경의 C/C++ 소스 코드 빌드
12210정성태4/20/202023067.NET Framework: 903. .NET Framework의 Strong-named 어셈블리 바인딩 (1) - app.config을 이용한 바인딩 리디렉션 [1]파일 다운로드1
12209정성태4/13/202018892오류 유형: 614. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우 (2)
12208정성태4/12/202017002Linux: 29. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우
12207정성태4/2/202017966스크립트: 19. Windows PowerShell의 NonInteractive 모드
12206정성태4/2/202019790오류 유형: 613. 파일 잠금이 바로 안 풀린다면? - The process cannot access the file '...' because it is being used by another process.
12205정성태4/2/202016727스크립트: 18. Powershell에서는 cmd.exe의 명령어를 지원하진 않습니다.
12204정성태4/1/202017366스크립트: 17. Powershell 명령어에 ';' (semi-colon) 문자가 포함된 경우
12203정성태3/18/202020326오류 유형: 612. warning: 'C:\ProgramData/Git/config' has a dubious owner: '...'.
... 61  62  63  64  65  66  67  68  69  [70]  71  72  73  74  75  ...