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

(시리즈 글이 10개 있습니다.)
.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




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)
12653정성태5/31/202110394.NET Framework: 1060. 닷넷 GC에 새롭게 구현되는 DPAD(Dynamic Promotion And Demotion for GC)
12652정성태5/31/20218493VS.NET IDE: 164. Visual Studio - Web Deploy로 Publish 시 암호창이 매번 뜨는 문제
12651정성태5/31/20218709오류 유형: 720. PostgreSQL - ERROR: 22P02: malformed array literal: "..."
12650정성태5/17/20218026기타: 82. OpenTabletDriver의 버튼에 더블 클릭을 매핑 및 게임에서의 지원 방법
12649정성태5/16/20219384.NET Framework: 1059. 세대 별 GC(Garbage Collection) 방식에서 Card table의 사용 의미 [1]
12648정성태5/16/20218005사물인터넷: 66. PC -> FTDI -> NodeMCU v1 ESP8266 기기를 UART 핀을 연결해 직렬 통신하는 방법파일 다운로드1
12647정성태5/15/20219270.NET Framework: 1058. C# - C++과의 연동을 위한 구조체의 fixed 배열 필드 사용파일 다운로드1
12646정성태5/15/20218368사물인터넷: 65. C# - Arduino IDE의 Serial Monitor 기능 구현파일 다운로드1
12645정성태5/14/20218077사물인터넷: 64. NodeMCU v1 ESP8266 - LittleFS를 이용한 와이파이 접속 정보 업데이트파일 다운로드1
12644정성태5/14/20219213오류 유형: 719. 윈도우 - 제어판의 "프로그램 및 기능" / "Windows 기능 켜기/끄기" 오류 0x800736B3
12643정성태5/14/20218409오류 유형: 718. 서버 유형의 COM+ 사용 시 0x80080005(Server execution failed) 오류 발생
12642정성태5/14/20219344오류 유형: 717. The 'Microsoft.ACE.OLEDB.12.0' provider is not registered on the local machine.
12641정성태5/13/20219032디버깅 기술: 179. 윈도우용 .NET Core 3 이상에서 Windbg의 sos 사용법
12640정성태5/13/202112003오류 유형: 716. RDP 연결 - Because of a protocol error (code: 0x112f), the remote session will be disconnected. [1]
12639정성태5/12/20218863오류 유형: 715. Arduino: Open Serial Monitor - The module '...\detection.node' was compiled against a different Node.js version using NODE_MODULE_VERSION
12638정성태5/12/20219789사물인터넷: 63. NodeMCU v1 ESP8266 - 펌웨어 내 파일 시스템(SPIFFS, LittleFS) 및 EEPROM 활용
12637정성태5/10/20219425사물인터넷: 62. NodeMCU v1 ESP8266 보드의 A0 핀에 다중 아날로그 센서 연결 [1]
12636정성태5/10/20219644사물인터넷: 61. NodeMCU v1 ESP8266 보드의 A0 핀 사용법 - FSR-402 아날로그 압력 센서 연동파일 다운로드1
12635정성태5/9/20218948기타: 81. OpenTabletDriver를 (관리자 권한으로 실행하지 않고도) 관리자 권한의 프로그램에서 동작하게 만드는 방법
12634정성태5/9/20218008개발 환경 구성: 572. .NET에서의 신뢰도 등급 조정 - 외부 Manifest 파일을 두는 방법파일 다운로드1
12633정성태5/7/20219485개발 환경 구성: 571. UAC - 관리자 권한 없이 UIPI 제약을 없애는 방법
12632정성태5/7/20219662기타: 80. (WACOM도 지원하는) Tablet 공통 디바이스 드라이버 - OpenTabletDriver
12631정성태5/5/20219579사물인터넷: 60. ThingSpeak 사물인터넷 플랫폼에 ESP8266 NodeMCU v1 + 조도 센서 장비 연동파일 다운로드1
12630정성태5/5/20219895사물인터넷: 59. NodeMCU v1 ESP8266 보드의 A0 핀 사용법 - CdS Cell(GL3526) 조도 센서 연동파일 다운로드1
12629정성태5/5/202111648.NET Framework: 1057. C# - CoAP 서버 및 클라이언트 제작 (UDP 소켓 통신) [1]파일 다운로드1
12628정성태5/4/20219621Linux: 39. Eclipse 원격 디버깅 - Cannot run program "gdb": Launching failed
... 31  32  33  34  35  36  37  38  [39]  40  41  42  43  44  45  ...