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

교재의 예제코드대로 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)
5617Edun2/25/20224349ArgumentOutOrRangeException에러 발생 [2]파일 다운로드1
5616csha...2/24/20224261readonly struct로 선언된 구조체를 특정 클래스에서 멤버변수로 가지는 경우 [1]
5615장성욱2/22/20226450SetThreadAffinityMask를 이용한 쓰레드 지정하는 방법에 대해 궁금합니다. [4]
5614민우2/22/20225811SSL 통신 문의 [6]
5613김인태2/22/20224299서버와 PC 간의 어플리케이션 성능 차이 [1]
5612팬입니다2/20/20224116Kastrel 서버 관련 [1]
5611차가워2/19/20224336stopWatch 늘어짐 문의 [3]
5610차가워2/18/20224221Stopwatch 늘어짐 문제 [1]
5609cs린이2/15/20224556c# 함수의 호출 방식에 대해 궁금합니다! [2]
5608지호2/10/20225088시작하세요 C# 8.0 중 제네릭타입의 IEnumerable [3]파일 다운로드1
5607이로운2/10/20224309안녕하세요. 궁금한게 있어서 질문드립니다. [1]
5605강성봉2/10/20224310TCP PSH flag 패킷 수신 에러 [1]
5604LW2/9/20224443VISUAL STUDIO 2019 ==> 2020 설치시 오류가 생겨서 문의드립니다. [3]
5603김진명2/9/20225736C# 10.0 출간은 언제쯤 계획하고 계신가요? [1]
5602신갑영2/8/20224542윈폼에 대해서 질문 드립니다. [1]
5601김인태2/4/20224771setup project 관련 [7]
5600itkim2/3/20225643윈도우 서버 계정 패스워드 인증 문의 [5]
5599레드골드2/3/20226144c#으로 ms word 제어 가능할까요? [6]
5598jaew...2/2/20224824Dictionary는 참조형식인가요?? [1]
5597재원2/2/20225162c# 9.0에 대한 내용을 받을 수 있나요? [1]
5596kss1/26/20224879듀얼 모니터 환경에서 wpf 프로그램 크기 변경 [2]
5594mira...1/25/20224531안녕하세요 try~catch 와 SuspendLayout~ResumeLayout 간 호출 문의드립니다! [2]
5593C#스터디1/12/20225456TaskAwaiter 구현 질문 입니다. [1]
5591유지킴12/24/20215054outofmemory in 32bit [2]파일 다운로드1
5590kss12/23/20214501포인터 메모리 에러 [1]
5589초급12/22/20214700c# -> 라즈베리파이(db 접속)시 에러 발생 [7]
1  2  3  4  5  6  7  8  9  10  11  12  [13]  14  15  ...