Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

(시리즈 글이 11개 있습니다.)
.NET Framework: 404. 리플렉션을 이용해 닷넷 LicenseManager를 우회할 수 있는 사례
; https://www.sysnet.pe.kr/2/0/1565

.NET Framework: 428. .NET Reflection으로 다차원/Jagged 배열을 구분하는 방법
; https://www.sysnet.pe.kr/2/0/1653

.NET Framework: 537. C# - Reflection의 박싱 없이 값 형식을 다루는 방법
; https://www.sysnet.pe.kr/2/0/10866

.NET Framework: 685. C# - 구조체(값 형식)의 필드를 리플렉션을 이용해 값을 바꾸는 방법
; https://www.sysnet.pe.kr/2/0/11312

.NET Framework: 785. public으로 노출되지 않은 다른 어셈블리의 delegate 인스턴스를 Reflection으로 생성하는 방법
; https://www.sysnet.pe.kr/2/0/11583

.NET Framework: 842. .NET Reflection을 대체할 System.Reflection.Metadata 소개
; https://www.sysnet.pe.kr/2/0/11930

.NET Framework: 924. C# - Reflection으로 변경할 수 없는 readonly 정적 필드
; https://www.sysnet.pe.kr/2/0/12256

.NET Framework: 1045. C# - 런타임 시점에 이벤트 핸들러를 만들어 Reflection을 이용해 구독하는 방법
; https://www.sysnet.pe.kr/2/0/12609

.NET Framework: 1046. C# - 컴파일 시점에 참조할 수 없는 타입을 포함한 이벤트 핸들러를 Reflection을 이용해 구독하는 방법
; https://www.sysnet.pe.kr/2/0/12610

닷넷: 2155. C# - .NET 8 런타임부터 (Reflection 없이) 특성을 이용해 public이 아닌 멤버 호출 가능
; https://www.sysnet.pe.kr/2/0/13436

닷넷: 2249. C# - 부모의 필드/프로퍼티에 대해 서로 다른 자식 클래스 간에 Reflection 접근이 동작할까요?
; https://www.sysnet.pe.kr/2/0/13608




C# - 런타임 시점에 이벤트 핸들러를 만들어 Reflection을 이용해 구독하는 방법

간단하게 이벤트 하나를 제공하는 타입을 만들어,

public class TempEventArgs : EventArgs
{
    public string Name;
}

public class MyTemp
{
    public void Create()
    {
        Created(this, new TempEventArgs { Name = "MyTEmp" });
    }

    public event EventHandler<TempEventArgs> Created;
}

Reflection을 이용하면 다음과 같이 이벤트 핸들러를 Reflection으로 추가할 수 있습니다.

using System;
using System.Reflection;

class Program
{
    static void Main(string[] args)
    {
        MyTemp instance = new MyTemp();
        MyTempEventTest(instance);

        instance.Create();
    }

    private static void Instance_Created(object sender, TempEventArgs e)
    {
        Console.WriteLine("Event Fired: " + e.Name);
    }

    private static void MyTempEventTest(object objTarget)
    {
        EventInfo ei = typeof(MyTemp).GetEvent("Created", BindingFlags.Public | BindingFlags.Instance);
        ei.AddEventHandler(objTarget, (EventHandler<TempEventArgs>)Instance_Created);
    }
}




여기까지만 하면 재미가 없으니, 상황을 좀 꼬아서 이벤트 핸들러까지 동적으로 런타임에 만들어 구독을 추가해 보겠습니다. 사실 이렇게까지 만들 일은 거의 없지만 그래도 DynamicMethod의 예제로써,

How to: Define and Execute Dynamic Methods
; https://learn.microsoft.com/en-us/dotnet/framework/reflection-and-codedom/how-to-define-and-execute-dynamic-methods

나름 괜찮으니 ^^ 이를 이용해 다음과 같이 IL 코드를 직접 추가해 이벤트 핸들러를 동적으로 만들 수 있습니다.

static void Main(string[] args)
{
    MyTemp instance = new MyTemp();

    SubscribeAtRuntime(instance);

    instance.Create();
}

private static void SubscribeAtRuntime(MyTemp instance)
{
    Type targetType = instance.GetType();
        
    EventInfo ei = targetType.GetEvent("Created", BindingFlags.Public | BindingFlags.Instance);
    MethodInfo consoleWriteMethod = typeof(Console).GetMethod("Write", BindingFlags.Public | BindingFlags.Static, null, new Type [] { typeof(object) }, null);
    MethodInfo consoleWriteLineMethod = typeof(Console).GetMethod("WriteLine", BindingFlags.Public | BindingFlags.Static, null, new Type[] { typeof(object) }, null);
    FieldInfo nameField = typeof(TempEventArgs).GetField("Name", BindingFlags.Public | BindingFlags.Instance);

    DynamicMethod proxyMethod = new DynamicMethod(
        "Proxy_Created",
        typeof(void), new Type[] { typeof(object), typeof(TempEventArgs) });

    ILGenerator il = proxyMethod.GetILGenerator();

    // Console.Write("Event Fired: " );
    il.Emit(OpCodes.Ldstr, "Event Fired: ");
    il.Emit(OpCodes.Call, consoleWriteMethod);

    // Console.WriteLine(e.Name);
    il.Emit(OpCodes.Ldarg_1);
    il.Emit(OpCodes.Ldfld, nameField);
    il.Emit(OpCodes.Call, consoleWriteLineMethod);
    il.Emit(OpCodes.Ret);

    Type eventHandlerType = typeof(EventHandler<TempEventArgs>);
    Delegate proxyDelegate = proxyMethod.CreateDelegate(eventHandlerType);

    ei.AddEventHandler(instance, proxyDelegate);
}

위에서 DynamicMethod.CreateDelegate 메서드에,

DynamicMethod.CreateDelegate Method
; https://learn.microsoft.com/en-us/dotnet/api/system.reflection.emit.dynamicmethod.createdelegate

전달한 타입은 "System.EventHandler`1[TempEventArgs]"입니다. 그리고 이 타입은 EventInfo 인스턴스를 통해서도 구할 수 있는데, 따라서 다음과 같이 전달하는 것도 가능합니다.

Delegate proxyDelegate = proxyMethod.CreateDelegate(ei.EventHandlerType);
ei.AddEventHandler(instance, proxyDelegate);

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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







[최초 등록일: ]
[최종 수정일: 11/7/2023]

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

비밀번호

댓글 작성자
 




... 31  32  33  34  [35]  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13063정성태5/18/202215624.NET Framework: 2015. C# - 인라인 메서드(inline methods)
13062정성태5/17/202216546.NET Framework: 2014. C# - async/await 그리고 스레드 (4) 비동기 I/O 재현 [1]파일 다운로드1
13061정성태5/16/202215325.NET Framework: 2013. C# - FILE_FLAG_OVERLAPPED가 적용된 파일의 읽기/쓰기 시 Position 관리파일 다운로드1
13060정성태5/15/202218802.NET Framework: 2012. C# - async/await 그리고 스레드 (3) Task.Delay 재현파일 다운로드1
13059정성태5/14/202216721.NET Framework: 2011. C# - CLR ThreadPool의 I/O 스레드에 작업을 맡기는 방법 [1]파일 다운로드1
13058정성태5/13/202216688.NET Framework: 2010. C# - ThreadPool.SetMaxThreads 사용법 [1]
13057정성태5/12/202218231오류 유형: 812. 파이썬 - ImportError: cannot import name ...
13056정성태5/12/202214333.NET Framework: 2009. C# - async/await 그리고 스레드 (2) MyTask의 호출 흐름 [2]파일 다운로드1
13055정성태5/11/202218735.NET Framework: 2008. C# - async/await 그리고 스레드 (1) MyTask로 재현 [11]파일 다운로드1
13054정성태5/11/202215622.NET Framework: 2007. C# - 10진수 숫자를 담은 문자열을 숫자로 변환하는 방법 [11]파일 다운로드1
13053정성태5/10/202215254.NET Framework: 2006. C# - GC.KeepAlive 메서드의 역할
13052정성태5/9/202215367.NET Framework: 2005. C# - 생성한 참조 개체가 언제 GC의 정리 대상이 될까요?
13051정성태5/8/202214731.NET Framework: 2004. C# XingAPI - ACF 검색 결과로 구한 CSV 파일을 통해 퀀트 종목 찾기파일 다운로드1
13050정성태5/6/202214997.NET Framework: 2003. C# - COM 개체의 이벤트 핸들러에서 발생하는 예외에 대한 CLR의 특별 대우파일 다운로드1
13049정성태5/6/202212832오류 유형: 811. GoLand - Error: Cannot find package
13048정성태5/6/202215017오류 유형: 810. "ASUS TUF GAMING B550M-PLUS (WI-FI)" 모델에서 블루투스 장치가 인식이 안 되는 문제
13047정성태5/6/202214566오류 유형: 809. Speech Recognition could not start
13046정성태5/5/202215104.NET Framework: 2002. C# XingAPI - ACF 파일을 이용한 퀀트 종목 찾기(t1857)
13045정성태5/5/202215418.NET Framework: 2001. C# XingAPI - 주식 종목에 따른 PBR, PER, ROE 구하는 방법(t3341 예제)
13044정성태5/4/202214174오류 유형: 808. error : clang++ exited with code 127
13043정성태5/3/202213300오류 유형: 807. C# - 닷넷 응용 프로그램에서 Informix DB 사용 시 오류 메시지 정리
13042정성태5/3/202214493.NET Framework: 2000. C# - 닷넷 응용 프로그램에서 Informix DB 사용 방법파일 다운로드1
13041정성태4/28/202214580개발 환경 구성: 642. Informix 데이터베이스 docker 환경 구성
13040정성태4/27/202214186VC++: 156. 비주얼 스튜디오 - Linux C/C++ 프로젝트에서 openssl 링크하는 방법
13039정성태4/27/202217305.NET Framework: 1999. C# - Playwright를 이용한 간단한 브라우저 제어 실습
13038정성태4/26/202213478오류 유형: 806. twine 실행 시 ConfigParser.ParsingError: File contains parsing errors: /root/.pypirc
... 31  32  33  34  [35]  36  37  38  39  40  41  42  43  44  45  ...