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)
12578정성태3/27/20219073개발 환경 구성: 560. Docker Desktop for Windows 기반의 Kubernetes 구성 (2) - WSL 2 인스턴스에 kind가 구성한 k8s 서비스 위치
12577정성태3/26/202111151개발 환경 구성: 559. Docker Desktop for Windows 기반의 Kubernetes 구성 - WSL 2 인스턴스에 kind 도구로 k8s 클러스터 구성
12576정성태3/25/20218932개발 환경 구성: 558. Docker Desktop for Windows에서 DockerDesktopVM 기반의 Kubernetes 구성 (2) - k8s 서비스 위치
12575정성태3/24/20218021개발 환경 구성: 557. Docker Desktop for Windows에서 DockerDesktopVM 기반의 Kubernetes 구성
12574정성태3/23/202111977.NET Framework: 1030. C# Socket의 Close/Shutdown 동작 (동기 모드)
12573정성태3/22/20219834개발 환경 구성: 556. WSL 인스턴스 초기 설정 명령어 [1]
12572정성태3/22/20219361.NET Framework: 1029. C# - GC 호출로 인한 메모리 압축(Compaction)을 확인하는 방법파일 다운로드1
12571정성태3/21/20218546오류 유형: 706. WSL 2 기반으로 "Enable Kubernetes" 활성화 시 초기화 실패 [1]
12570정성태3/19/202112880개발 환경 구성: 555. openssl - CA로부터 인증받은 새로운 인증서를 생성하는 방법
12569정성태3/18/202111732개발 환경 구성: 554. WSL 인스턴스 export/import 방법 및 단축 아이콘 설정 방법
12568정성태3/18/20217370오류 유형: 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/20218742개발 환경 구성: 553. Docker Desktop for Windows를 위한 k8s 대시보드 활성화 [1]
12566정성태3/17/20219065개발 환경 구성: 552. Kubernetes - kube-apiserver와 REST API 통신하는 방법 (Docker Desktop for Windows 환경)
12565정성태3/17/20216560오류 유형: 704. curl.exe 실행 시 dll not found 오류
12564정성태3/16/20217044VS.NET IDE: 160. 새 프로젝트 창에 C++/CLI 프로젝트 템플릿이 없는 경우
12563정성태3/16/20218980개발 환경 구성: 551. C# - JIRA REST API 사용 정리 (3) jira-oauth-cli 도구를 이용한 키 관리
12562정성태3/15/202110111개발 환경 구성: 550. C# - JIRA REST API 사용 정리 (2) JIRA OAuth 토큰으로 API 사용하는 방법파일 다운로드1
12561정성태3/12/20218715VS.NET IDE: 159. Visual Studio에서 개행(\n, \r) 등의 제어 문자를 치환하는 방법 - 정규 표현식 사용
12560정성태3/11/202110064개발 환경 구성: 549. ssh-keygen으로 생성한 개인키/공개키 파일을 각각 PKCS8/PEM 형식으로 변환하는 방법
12559정성태3/11/20219462.NET Framework: 1028. 닷넷 5 환경의 Web API에 OpenAPI 적용을 위한 NSwag 또는 Swashbuckle 패키지 사용 [2]파일 다운로드1
12558정성태3/10/20218945Windows: 192. Power Automate Desktop (Preview) 소개 - Bitvise SSH Client 제어 [1]
12557정성태3/10/20217602Windows: 191. 탐색기의 보안 탭에 있는 "Object name" 경로에 LEFT-TO-RIGHT EMBEDDING 제어 문자가 포함되는 문제
12556정성태3/9/20216887오류 유형: 703. PowerShell ISE의 Debug / Toggle Breakpoint 메뉴가 비활성 상태인 경우
12555정성태3/8/20218917Windows: 190. C# - 레지스트리에 등록된 DigitalProductId로부터 라이선스 키(Product Key)를 알아내는 방법파일 다운로드2
12554정성태3/8/20218755.NET Framework: 1027. 닷넷 응용 프로그램을 위한 PDB 옵션 - full, pdbonly, portable, embedded
12553정성태3/5/20219219개발 환경 구성: 548. 기존 .NET Framework 프로젝트를 .NET Core/5+ 용으로 변환해 주는 upgrade-assistant, try-convert 도구 소개 [4]
... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...