Microsoft MVP성태의 닷넷 이야기
교재 199page 델리게이트와 object를 이용한 범용 정렬 코드 [링크 복사], [링크+제목 복사]
조회: 9044
글쓴 사람
익명
홈페이지
첨부 파일
 

교재의 예제코드대로 delegate를 이용해서 다양한 타입을 받을 수 있는 범용 정렬코드를 작성하다가 문제가 생겼습니다.
분명 int도 System.ValueType을 상속받았고 System.ValueType은 object를 상속받았으니 형변환이 될텐데, 단일 변수의 경우는 되지만 배열의 경우에는 안됩니다.
AscSortByNumber함수의 내용처럼 단일 변수일 경우에는 형변환이 되는데 배열일 때는 안되는 이유를 모르겠습니다.
1)안되는 이유와
2)해결방법
이 있다면 해결방법도 알고싶습니다.

아래의 코드는 교재의 예제코드를 숫자값들도 정렬할 수 있게끔 제가 마저 작성한 코드이며
이슈되는 코드들 위에 주석을 달았습니다.



using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace ConsoleApp3
{
    delegate bool CompareDelegate(object arg1, object arg2);

    class SortObject
    {
        object[] things;

        //object[]가 int[]는 받을 수가 없습니다.
        public SortObject(object[] things)
        {
            this.things = things;
        }

        public void Sort(CompareDelegate compareMethod)
        {
            object temp;

            for (int i = 0; i < things.Length; i++)
            {
                int lowPos = i;

                for (int j = i + 1; j < things.Length; j++)
                {
                    if (compareMethod(things[j], things[lowPos]))
                    {
                        lowPos = j;
                    }
                }

                temp = things[lowPos];
                things[lowPos] = things[i];
                things[i] = temp;
            }
        }

        public void Display()
        {
            for (int i = 0; i < things.Length; i++)
            {
                Console.WriteLine(things[i] + ", ");
            }
        }
    }

    class Person
    {
        public int Age;
        public string Name;

        public Person(int age, string name)
        {
            this.Age = age;
            this.Name = name;
        }

        public override string ToString()
        {
            return Name + ": " + Age;
        }
    }

    class Program
    {
        static bool AscSortByName(object arg1, object arg2)
        {
            Person person1 = arg1 as Person;
            Person person2 = arg2 as Person;

            return person1.Name.CompareTo(person2.Name) < 0;
        }

        static bool AscSortByNumber(object arg1, object arg2)
        {
            //이처럼 배열이 아닌 경우에는 잘됩니다.
            int number1 = (int)arg1;
            int number2 = (int)arg2;

            return number1 < number2;
        }

        static void Main(string[] args)
        {
            Person[] people = new Person[] { new Person(30, "홍길동"), new Person(17, "김삿갓"), new Person(24, "가나연") };

            SortObject sortPerson = new SortObject(people);

            sortPerson.Sort(AscSortByName);
            sortPerson.Display();


            Console.WriteLine();


            int[] intArray = new int[] { 5, 2, 3, 1, 0, 4 };

            //int[]를 object[]로 변환할 수 없습니다.
            //(object[])intArray, intArray as object[] 모두 안됩니다.
            SortObject sortNumber = new SortObject(intArray);

            sortNumber.Sort(AscSortByNumber);
            sortNumber.Display();
        }
    }
}









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


비밀번호

댓글 작성자
 



2018-03-12 03시44분
사실, C# 컴파일러가 그런 부분에 대해 처리를 잘 해줬을 뿐이지 원래는 값 형식과 참조 형식 간의 그런 변화는 가능하지 않습니다. (실제로 자바는 int와 Integer 타입이 구분되어 있죠!)

C#의 경우 그런 과정을 개발자에게는 투명하게 box/unbox 해주기 때문에 그런 오해가 발생할 수 있습니다. 즉 int 값을 object에 전달하면 원래는 불가능하지만 C# 컴파일러는 그 과정에 box IL 코드를 자동으로 넣어 처리를 합니다.

배열의 경우에는 더욱 문제가 복잡해지는데요, 배열은 각각의 타입이 모두 독자적인 참조형으로 구현됩니다. 즉 int [] 배열은 값 형식을 담고 있지만 구현 자체는 참조 형식입니다. object []의 경우에도 요소가 object 인 별도의 object 배열을 담고 있는 참조 형식으로 구현됩니다. 따라서 독자적인 배열들이 참조 형식이고 그것의 공통 부모는 단지 object 타입일 뿐입니다. 즉, object[]를 상속받아 int[]가 구현되는 것은 아닙니다.

참고로, 원하시는 그런 구조를 매끄럽게 구현하려면 generic을 사용해야 합니다.
정성태

1  2  3  4  5  6  7  8  9  [10]  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5699갓경섭7/29/20224438C# 프로그램 느려짐 현상 [2]
5698kss7/29/20225203ValueTask 질문입니다. [5]
5697스팸7/28/20225188윈도우서버 Port 고갈문제 (Event ID 4227, 4231) [3]파일 다운로드1
5696유석7/27/20225472닷넷 4.X 와 닷넷 6.0 버전 차이 [6]
5695초보코더7/26/20224835[WCF] EndpointNotFoundException 기본 연결이 닫혔습니다. 원격 서버에 연결할 수 없습니다. [1]
5693차가워7/22/20224308socket 종료 시 reveive수신부에서 에러 나는거 처리 문의 [5]
5692도비7/21/20223918Ubuntu LVM 확장 관련 [1]파일 다운로드1
5691따봉이7/21/20224612Task 비동기 UI 크로스 쓰레딩 Err 관련 [1]
5690훈이7/20/20223852wpf질문입니다 [2]
5689질문7/20/20223645비동기 함수를 사용하는 방법에 대한 질문입니다 [1]
5688김정현7/11/20223735소스 디버깅(F5)시 프로그램 켜지지 않고 디버깅 가능한 상태 [1]
5687규철7/6/20223853개발업무 문서 관리 방법 문의 [3]
5686권용완7/5/20223821Linq Entites 에서 string[]에 담긴 데이터를 제외하는 방법 문의 [3]
5685차가워7/4/20223991메모리 선택 문의 [1]
5683김민혁7/1/20224542HttpWebRequest POST 전송 관련해서 질문 드립니다. [4]파일 다운로드1
5682김기헌6/20/20223984안녕하세요 서로 다른 스레드에서 동일 인스턴스의 메소드를 호출할때 질문드립니다 [2]
5681미래는투...6/19/20223870VC++ 빌드 관련 문의 (h파일이 바뀌었는데 컴파일을 꼭 하지 않아도 되나요?) [1]
5680Syong6/12/20224873EXE파일에 속성값 추가하는 방법이 있나요? [2]파일 다운로드1
5679차가워6/8/20224006tcpip socket 여러 랜카드 사용 시 연결 문의 [1]
5678차가워6/7/20223917원자적 실행 문의 [3]
5677장성욱6/7/20223940블루투스 포트와 시리얼포트 연결시 에러 [2]
5675차가워5/29/20224622윈도우설정 프로세스사용계획 옵션과 c# 스레드 [1]
5674민우5/29/20224494C++ 의 RTTI, Reflection 으로 객체 멤버변수, 멤버함수 목록 추출 문의 [2]
5673조은현5/25/20225427안녕하세요 선생님! c#에서 TCP/IP의 keep alive 기능의 사용법에 대해서 질문드리고 싶어요! [2]파일 다운로드2
5672game...5/23/20225324안녕하세요 GC를 일시적으로 중단시키는 방법이 있을까요? [5]
5671한예지 donator5/20/20224927델리게이트와 함수포인터 선언 시, 차이점 질문 있습니다. [3]
1  2  3  4  5  6  7  8  9  [10]  11  12  13  14  15  ...