Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 4개 있습니다.)
.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

닷넷: 2251. C# - 제네릭 인자를 가진 타입을 생성하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/13610




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]

... 61  62  63  64  65  66  67  [68]  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12236정성태6/19/202018429오류 유형: 621. .NET Standard 대상으로 빌드 시 dynamic 예약어에서 컴파일 오류 - error CS0656: Missing compiler required member 'Microsoft.CSharp.RuntimeBinder.CSharpArgumentInfo.Create'
12235정성태6/19/202017598오류 유형: 620. Windows 10 - Inaccessible boot device 블루 스크린
12234정성태6/19/202016969개발 환경 구성: 494. NuGet - nuspec의 패키지 스키마 버전(네임스페이스) 업데이트 방법
12233정성태6/19/202017508오류 유형: 619. SQL 서버 - The transaction log for database '...' is full due to 'LOG_BACKUP'. - 두 번째 이야기
12232정성태6/19/202016205오류 유형: 618. SharePoint - StoreBusyRetryLater 오류
12231정성태6/15/202019480.NET Framework: 911. Console/Service Application을 위한 SynchronizationContext - AsyncContext
12230정성태6/15/202018341오류 유형: 617. IMetaDataImport::GetMethodProps가 반환하는 IL 코드 주소(RVA) 문제
12229정성태6/13/202020191.NET Framework: 910. USB/IP PROJECT를 이용해 C#으로 USB Keyboard + Mouse 가상 장치 만들기 [1]
12228정성태6/12/202019540.NET Framework: 909. C# - Source Generator를 적용한 XmlCodeGenerator파일 다운로드1
12227정성태6/12/202023472오류 유형: 616. Visual Studio의 느린 업데이트 속도에 대한 원인 분석 [5]
12226정성태6/11/202021543개발 환경 구성: 493. OpenVPN의 네트워크 구성 [4]파일 다운로드1
12225정성태6/11/202019474개발 환경 구성: 492. 윈도우에 OpenVPN 설치 - 클라이언트 측 구성
12224정성태6/11/202028221개발 환경 구성: 491. 윈도우에 OpenVPN 설치 - 서버 측 구성 [1]
12223정성태6/9/202023902.NET Framework: 908. C# - Source Generator 소개 [10]파일 다운로드2
12222정성태6/3/202017385VS.NET IDE: 146. error information: "CryptQueryObject" (-2147024893/0x80070003)
12221정성태6/3/202017135Windows: 170. 비어 있지 않은 디렉터리로 symbolic link(junction) 연결하는 방법
12220정성태6/3/202020887.NET Framework: 907. C# DLL로부터 TLB 및 C/C++ 헤더 파일(TLH)을 생성하는 방법
12219정성태6/1/202019722.NET Framework: 906. C# - lock (this), lock (typeof(...))를 사용하면 안 되는 이유파일 다운로드1
12218정성태5/27/202019144.NET Framework: 905. C# - DirectX 게임 클라이언트 실행 중 키보드 입력을 감지하는 방법 [3]
12217정성태5/24/202017202오류 유형: 615. Transaction count after EXECUTE indicates a mismatching number of BEGIN and COMMIT statements. Previous count = 0, current count = 1.
12216정성태5/15/202020761.NET Framework: 904. USB/IP PROJECT를 이용해 C#으로 USB Keyboard 가상 장치 만들기 [14]파일 다운로드1
12215정성태5/12/202026702개발 환경 구성: 490. C# - (Wireshark의) USBPcap을 이용한 USB 패킷 모니터링 [10]파일 다운로드1
12214정성태5/5/202018419개발 환경 구성: 489. 정식 인증서가 있는 경우 Device Driver 서명하는 방법 (2) - UEFI/SecureBoot [1]
12213정성태5/3/202019544개발 환경 구성: 488. (User-mode 코드로 가상 USB 장치를 만들 수 있는) USB/IP PROJECT 소개
12212정성태5/1/202016751개발 환경 구성: 487. UEFI / Secure Boot 상태인지 확인하는 방법
12211정성태4/27/202019300개발 환경 구성: 486. WSL에서 Makefile로 공개된 리눅스 환경의 C/C++ 소스 코드 빌드
... 61  62  63  64  65  66  67  [68]  69  70  71  72  73  74  75  ...