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]

1  2  3  4  5  6  7  8  9  [10]  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13728정성태9/7/20248489C/C++: 172. Windows - C 런타임에서 STARTUPINFO의 cbReserved2, lpReserved2 멤버를 사용하는 이유파일 다운로드1
13727정성태9/6/20249157개발 환경 구성: 722. ARM 플랫폼 빌드를 위한 미니 PC(?) - Khadas VIM4 [1]
13726정성태9/5/20248204C/C++: 171. C/C++ - 윈도우 운영체제에서의 file descriptor와 HANDLE파일 다운로드1
13725정성태9/4/20246798디버깅 기술: 201. WinDbg - sos threads 명령어 실행 시 "Failed to request ThreadStore"
13724정성태9/3/20249646닷넷: 2296. Win32/C# - 자식 프로세스로 HANDLE 상속파일 다운로드1
13723정성태9/2/20249053C/C++: 170. Windows - STARTUPINFO의 cbReserved2, lpReserved2 멤버 사용자 정의파일 다운로드2
13722정성태9/2/20246884C/C++: 169. C/C++ - CRT(C Runtime) 함수에 의존성이 없는 프로젝트 생성
13721정성태8/30/20246673C/C++: 168. Visual C++ CRT(C Runtime DLL: msvcr...dll)에 대한 의존성 제거 - 두 번째 이야기
13720정성태8/29/20247039VS.NET IDE: 193. C# - Visual Studio의 자식 프로세스 디버깅
13719정성태8/28/20247063Linux: 79. C++ - pthread_mutexattr_destroy가 없다면 메모리 누수가 발생할까요?
13718정성태8/27/20248908오류 유형: 921. Visual C++ - error C1083: Cannot open include file: 'float.h': No such file or directory [2]
13717정성태8/26/20248488VS.NET IDE: 192. Visual Studio 2022 - Windows XP / 2003용 C/C++ 프로젝트 빌드
13716정성태8/21/20247655C/C++: 167. Visual C++ - 윈도우 환경에서 _execv 동작 [1]
13715정성태8/19/20248839Linux: 78. 리눅스 C/C++ - 특정 버전의 glibc 빌드 (docker-glibc-builder)
13714정성태8/19/20247424닷넷: 2295. C# 12 - 기본 생성자(Primary constructors) (책 오타 수정) [3]
13713정성태8/16/20248956개발 환경 구성: 721. WSL 2에서의 Hyper-V Socket 연동
13712정성태8/14/20248257개발 환경 구성: 720. Synology NAS - docker 원격 제어를 위한 TCP 바인딩 추가
13711정성태8/13/20248751Linux: 77. C# / Linux - zombie process (defunct process) [1]파일 다운로드1
13710정성태8/8/20249325닷넷: 2294. C# 13 - (6) iterator 또는 비동기 메서드에서 ref와 unsafe 사용을 부분적으로 허용파일 다운로드1
13709정성태8/7/20248670닷넷: 2293. C# - safe/unsafe 문맥에 대한 C# 13의 (하위 호환을 깨는) 변화파일 다운로드1
13708정성태8/7/20248889개발 환경 구성: 719. ffmpeg / YoutubeExplode - mp4 동영상 파일로부터 Audio 파일 추출
13707정성태8/6/20248883닷넷: 2292. C# - 자식 프로세스의 출력이 4,096보다 많은 경우 Process.WaitForExit 호출 시 hang 현상파일 다운로드1
13706정성태8/5/20248626개발 환경 구성: 718. Hyper-V - 리눅스 VM에 새로운 디스크 추가
13705정성태8/4/20248867닷넷: 2291. C# 13 - (5) params 인자 타입으로 컬렉션 허용 [2]파일 다운로드1
13704정성태8/2/20249564닷넷: 2290. C# - 간이 dotnet-dump 프로그램 만들기파일 다운로드1
13703정성태8/1/20248140닷넷: 2289. "dotnet-dump ps" 명령어가 닷넷 프로세스를 찾는 방법
1  2  3  4  5  6  7  8  9  [10]  11  12  13  14  15  ...