Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 3개 있습니다.)
.NET Framework: 784. C# - 제네릭 인자를 가진 타입을 생성하는 방법
; https://www.sysnet.pe.kr/2/0/11582

.NET Framework: 787. object로 형변환된 인스턴스를 원래의 타입 인자로 제네릭 메서드를 호출하는 방법
; https://www.sysnet.pe.kr/2/0/11589

닷넷: 2145. C# - 제네릭의 형식 매개변수에 속한 (매개변수를 가진) 생성자를 호출하는 방법
; https://www.sysnet.pe.kr/2/0/13417




object로 형변환된 인스턴스를 원래의 타입 인자로 제네릭 메서드를 호출하는 방법

말이 좀 어려우니, 코드로 한번 볼까요? ^^

using System;

namespace ConsoleApp1
{
    enum MyEnum
    {
        A,
        B,
    }

    class Program
    {
        static void Main(string[] args)
        {
            object objValue = MyEnum.A;

            Program pg = new Program();
            pg.Test(objValue);
            pg.Test(MyEnum.A);
        }

        void Test<T>(T objValue)
        {
            Type type = typeof(T);
            Console.WriteLine(type.FullName); 
        }
    }
}

/*
출력 결과:

System.Object
ConsoleApp1.MyEnum
*/

위의 코드에서 같은 값임에도 불구하고 object로 형변환된 경우 제네릭 메서드가 "Test<object>(object objValue)"로 선택된 반면, 원래의 값으로 호출한 경우에는 "Test<ConsoleApp1.MyEnum>(ConsoleApp1.MyEnum objValue)"로 선택됩니다.

그러니까, object로도 "Test<ConsoleApp1.MyEnum>(ConsoleApp1.MyEnum objValue)"로 호출하고 싶다면 어떻게 해야 할까요? 어쩔 수 없습니다. 이런 경우에는 Reflection을 이용해야 하는데, 방식은 지난번 설명했던 내용을 기반으로 합니다.

C# - 제네릭 인자를 가진 타입을 생성하는 방법
; https://www.sysnet.pe.kr/2/0/11582

단지 이번에는 그 대상이 타입이 아니라 메서드로 바뀐 것뿐입니다.

using System;
using System.Reflection;

namespace ConsoleApp1
{
    enum MyEnum
    {
        A,
        B,
    }

    class Program
    {
        static void Main(string[] args)
        {
            object objValue = MyEnum.A;

            Program pg = new Program();
            pg.Test(objValue);
            pg.Test(MyEnum.A);

            pg.CallTest(objValue);
        }

        void CallTest(object objValue)
        {
            Type argType = objValue.GetType();

            MethodInfo mi = typeof(Program).GetMethod("Test", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
            MethodInfo testMethod = mi.MakeGenericMethod(new Type[] { argType });

            testMethod.Invoke(this, new object[] { objValue });
        }

        void Test<T>(T objValue)
        {
            Type type = typeof(T);
            Console.WriteLine(type.FullName);
        }
    }
}

/*
출력 결과:

System.Object
ConsoleApp1.MyEnum
ConsoleApp1.MyEnum
*/

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




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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2018-07-07 08시03분
[guest] 질문에 대한 답변 고맙습니다.
아직 C#과 객체지향이 미숙하여
시간을 두고 천천히 곱씹으면서 코드를 음미해야하
겨우 이해할것같습니다.
이렇게 섬세하고도 자세한 답변까지 기대하지도 않았지만
매번 이렇게 답변해주시니



namespace ThankYou
{

  static class TextData
  {

    public static int[,] 고 { get; set; } = new int[,]
    {
      {2, 18, 0},
      {2, 18, 0},
      {16, 4, 0},
      {16, 4, 0},
      {10, 2, 0},
      {10, 2, 0},
      {0, 22, 0},
      {0, 22, 0}
    };


    public static int[,] 맙 { get; set; } = new int[,]
    {
      {20, 3, 0},

      {2, 16, 1},
      {2, 3, 0},

      {2, 16, 1},
      {2, 3, 0},
      
      {2, 3, 1},
      {10, 3, 1},
      {2, 3, 0},

      {2, 3, 1},
      {10, 3, 1},
      {2, 3, 0},

      {2, 3, 1},
      {10, 3, 1},
      {2, 6, 0},

      {2, 3, 1},
      {10, 3, 1},
      {2, 6, 0},

      {2, 3, 1},
      {10, 3, 1},
      {2, 3, 0},

      {2, 16, 1},
      {2, 3, 0},

      {2, 16, 1},
      {2, 3, 0},

      {20, 3, 0},
      {0, 0, 0},

      {6, 3, 1},
      {12, 3, 0},
      {6, 3, 1},
      {12, 3, 0},

      {6, 18, 0},
      {6, 18, 0},

      {6, 3, 1},
      {12, 3, 0},


      {6, 18, 0},
      {6, 18, 0},

    };

    public static int[,] 습 { get; set; } = new int[,]
    {
      {12, 3, 0},
      {10, 3, 1},
      {1, 3, 0},

      {8, 3, 1},
      {5, 3, 0},

      {6, 3, 1},
      {9, 3, 0},

      {0, 0, 0},

      {2, 24, 0},
      {2, 24, 0},

      {0, 0, 0},

      {5, 3, 1},
      {12, 3, 0},
      {5, 3, 1},
      {12, 3, 0},

      {5, 18, 0},
      {5, 18, 0},

      {5, 3, 1},
      {12, 3, 0},


      {5, 18, 0},
      {5, 18, 0}
    };

    public static int[,] 니 { get; set; } = new int[,]
    {
      {14, 3, 0},

      {2, 3, 1},
      {9, 3, 0},
      {2, 3, 1},
      {9, 3, 0},
      {2, 3, 1},
      {9, 3, 0},

      {2, 3, 1},
      {9, 3, 0},
      {2, 3, 1},
      {9, 3, 0},

      {2, 15, 0},
      {2, 15, 0},

      {14, 3, 0},
      {14, 3, 0},
      {14, 3, 0}
    };


    public static int[,] 다 { get; set; } = new int[,]
    {
      {20, 3, 0},

      {2, 15, 1},
      {3, 3, 0},

      {2, 15, 1},
      {3, 3, 0},

      {2, 3, 1},
      {15, 3, 0},

      {2, 3, 1},
      {15, 3, 0},

      {2, 3, 1},
      {15, 6, 0},

      {2, 3, 1},
      {15, 6, 0},

      {2, 3, 1},
      {15, 3, 0},

      {2, 16, 1},
      {2, 3, 0},

      {2, 16, 1},
      {2, 3, 0},

      {20, 3, 0},
      {20, 3, 0},



    };


  }

  class CreateConsoleText
  {

    void PrintLine(int empty = 0, int length = 1, int newLine = 0)
    {
      int limit = empty == 0 ? length : empty + length;
      for (int i = 0; i < limit; i++)
      {
        if(empty == 0)
        {
          Console.Write('.');
        }
        else
        {
          if(i < empty)
          {
            Console.Write(' ');
          }
          else
          {
            Console.Write('.');
          }
        }
        
        Thread.Sleep(10);
      }
      if (newLine == 0)
      {
        Console.WriteLine();
      }
    }

    void PintText(int[,] textData)
    {
      for (int i = 0; i < textData.Length / 3; i++)
      {
        PrintLine(textData[i, 0], textData[i, 1], textData[i, 2]);
      }

      for(int i=0; i<50; i++)
      {
        Console.Write('/');
      }
      Console.WriteLine();
    }

    public void start()
    {


      PintText(TextData.고);
      PintText(TextData.맙);
      PintText(TextData.습);
      PintText(TextData.니);
      PintText(TextData.다);


    }
  }


  class Program
  {
    static void Main(string[] args)
    {

      new CreateConsoleText().start();

      Console.ReadLine();
    }
  }
}

[guest]
2018-07-07 11시03분
[ilyl] 와우!
[guest]

... 31  32  33  34  35  36  37  38  39  40  41  42  [43]  44  45  ...
NoWriterDateCnt.TitleFile(s)
12546정성태3/3/20218338개발 환경 구성: 545. github workflow/actions에서 빌드시 snk 파일 다루는 방법 - Encrypted secrets
12545정성태3/2/202111061.NET Framework: 1026. 닷넷 5에 추가된 POH (Pinned Object Heap) [10]
12544정성태2/26/202111223.NET Framework: 1025. C# - Control의 Invalidate, Update, Refresh 차이점 [2]
12543정성태2/26/20219654VS.NET IDE: 158. C# - 디자인 타임(design-time)과 런타임(runtime)의 코드 실행 구분
12542정성태2/20/202111964개발 환경 구성: 544. github repo의 Release 활성화 및 Actions를 이용한 자동화 방법 [1]
12541정성태2/18/20219232개발 환경 구성: 543. 애저듣보잡 - Github Workflow/Actions 소개
12540정성태2/17/20219534.NET Framework: 1024. C# - Win32 API에 대한 P/Invoke를 대신하는 Microsoft.Windows.CsWin32 패키지
12539정성태2/16/20219427Windows: 189. WM_TIMER의 동작 방식 개요파일 다운로드1
12538정성태2/15/20219847.NET Framework: 1023. C# - GC 힙이 아닌 Native 힙에 인스턴스 생성 - 0SuperComicLib.LowLevel 라이브러리 소개 [2]
12537정성태2/11/202110791.NET Framework: 1022. UI 요소의 접근은 반드시 그 UI를 만든 스레드에서! - 두 번째 이야기 [2]
12536정성태2/9/20219843개발 환경 구성: 542. BDP(Bandwidth-delay product)와 TCP Receive Window
12535정성태2/9/20218965개발 환경 구성: 541. Wireshark로 확인하는 LSO(Large Send Offload), RSC(Receive Segment Coalescing) 옵션
12534정성태2/8/20219588개발 환경 구성: 540. Wireshark + C/C++로 확인하는 TCP 연결에서의 closesocket 동작 [1]파일 다운로드1
12533정성태2/8/20219231개발 환경 구성: 539. Wireshark + C/C++로 확인하는 TCP 연결에서의 shutdown 동작파일 다운로드1
12532정성태2/6/20219761개발 환경 구성: 538. Wireshark + C#으로 확인하는 ReceiveBufferSize(SO_RCVBUF), SendBufferSize(SO_SNDBUF) [3]
12531정성태2/5/20218718개발 환경 구성: 537. Wireshark + C#으로 확인하는 PSH flag와 Nagle 알고리듬파일 다운로드1
12530정성태2/4/202112848개발 환경 구성: 536. Wireshark + C#으로 확인하는 TCP 통신의 Receive Window
12529정성태2/4/20219965개발 환경 구성: 535. Wireshark + C#으로 확인하는 TCP 통신의 MIN RTO [1]
12528정성태2/1/20219346개발 환경 구성: 534. Wireshark + C#으로 확인하는 TCP 통신의 MSS(Maximum Segment Size) - 윈도우 환경
12527정성태2/1/20219585개발 환경 구성: 533. Wireshark + C#으로 확인하는 TCP 통신의 MSS(Maximum Segment Size) - 리눅스 환경파일 다운로드1
12526정성태2/1/20217459개발 환경 구성: 532. Azure Devops의 파이프라인 빌드 시 snk 파일 다루는 방법 - Secure file
12525정성태2/1/20217163개발 환경 구성: 531. Azure Devops - 파이프라인 실행 시 빌드 이벤트를 생략하는 방법
12524정성태1/31/20218224개발 환경 구성: 530. 기존 github 프로젝트를 Azure Devops의 빌드 Pipeline에 연결하는 방법 [1]
12523정성태1/31/20218218개발 환경 구성: 529. 기존 github 프로젝트를 Azure Devops의 Board에 연결하는 방법
12522정성태1/31/20219711개발 환경 구성: 528. 오라클 클라우드의 리눅스 VM - 9000 MTU Jumbo Frame 테스트
12521정성태1/31/20219753개발 환경 구성: 527. 이더넷(Ethernet) 환경의 TCP 통신에서 MSS(Maximum Segment Size) 확인 [1]
... 31  32  33  34  35  36  37  38  39  40  41  42  [43]  44  45  ...