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]

... 46  47  48  49  50  51  52  53  [54]  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12590정성태4/9/202116900.NET Framework: 1033. C# - .NET 4.0 이하에서 Console.IsInputRedirected 구현 [1]
12589정성태4/8/202118129.NET Framework: 1032. C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법 [1]
12588정성태4/7/202119861개발 환경 구성: 565. PowerShell - New-SelfSignedCertificate를 사용해 CA 인증서 생성 및 인증서 서명 방법
12587정성태4/6/202121138개발 환경 구성: 564. Windows 10 - ClickOnce 배포처럼 사용할 수 있는 MSIX 설치 파일 [1]
12586정성태4/5/202118007오류 유형: 710. Windows - Restart-Computer / shutdown 명령어 수행 시 Access is denied(E_ACCESSDENIED)
12585정성태4/5/202116977개발 환경 구성: 563. 기본 생성된 kubeconfig 파일의 내용을 새롭게 생성한 인증서로 구성하는 방법
12584정성태4/1/202118161개발 환경 구성: 562. kubeconfig 파일 없이 kubectl 옵션만으로 실행하는 방법
12583정성태3/29/202119049개발 환경 구성: 561. kubectl 수행 시 다른 k8s 클러스터로 접속하는 방법
12582정성태3/29/202118453오류 유형: 709. Visual C++ - 컴파일 에러 error C2059: syntax error: '__stdcall'
12581정성태3/28/202118410.NET Framework: 1031. WinForm/WPF에서 Console 창을 띄워 출력하는 방법 (2) - Output 디버깅 출력을 AllocConsole로 우회 [2]
12580정성태3/28/202116266오류 유형: 708. SQL Server Management Studio - Execution Timeout Expired.
12579정성태3/28/202116874오류 유형: 707. 중첩 가상화(Nested Virtualization) - The virtual machine could not be started because this platform does not support nested virtualization.
12578정성태3/27/202117317개발 환경 구성: 560. Docker Desktop for Windows 기반의 Kubernetes 구성 (2) - WSL 2 인스턴스에 kind가 구성한 k8s 서비스 위치
12577정성태3/26/202118946개발 환경 구성: 559. Docker Desktop for Windows 기반의 Kubernetes 구성 - WSL 2 인스턴스에 kind 도구로 k8s 클러스터 구성
12576정성태3/25/202116988개발 환경 구성: 558. Docker Desktop for Windows에서 DockerDesktopVM 기반의 Kubernetes 구성 (2) - k8s 서비스 위치
12575정성태3/24/202115567개발 환경 구성: 557. Docker Desktop for Windows에서 DockerDesktopVM 기반의 Kubernetes 구성 [1]
12574정성태3/23/202121099.NET Framework: 1030. C# Socket의 Close/Shutdown 동작 (동기 모드)
12573정성태3/22/202118517개발 환경 구성: 556. WSL 인스턴스 초기 설정 명령어 [1]
12572정성태3/22/202117813.NET Framework: 1029. C# - GC 호출로 인한 메모리 압축(Compaction)을 확인하는 방법파일 다운로드1
12571정성태3/21/202115827오류 유형: 706. WSL 2 기반으로 "Enable Kubernetes" 활성화 시 초기화 실패 [1]
12570정성태3/19/202121132개발 환경 구성: 555. openssl - CA로부터 인증받은 새로운 인증서를 생성하는 방법
12569정성태3/18/202121496개발 환경 구성: 554. WSL 인스턴스 export/import 방법 및 단축 아이콘 설정 방법
12568정성태3/18/202114824오류 유형: 705. C# 빌드 - Couldn't process file ... due to its being in the Internet or Restricted zone or having the mark of the web on the file.
12567정성태3/17/202116878개발 환경 구성: 553. Docker Desktop for Windows를 위한 k8s 대시보드 활성화 [1]
12566정성태3/17/202116702개발 환경 구성: 552. Kubernetes - kube-apiserver와 REST API 통신하는 방법 (Docker Desktop for Windows 환경)
12565정성태3/17/202113488오류 유형: 704. curl.exe 실행 시 dll not found 오류
... 46  47  48  49  50  51  52  53  [54]  55  56  57  58  59  60  ...