Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 6개 있습니다.)
Dictionary.Get(A) 대신 Dictionary.Get(A.GetHashCode())를 사용해서는 안 되는 이유


최근에 "Jeffrey Richter"가 쓴 "CLR via C#" (송기수 MVP 님이 번역하신 바로 그 책! 3rd edition이 새롭게 나왔다는 데 이번에도 송기수 님이 번역하셨으면 좋겠습니다. ^^) 책을 보다가 다음과 같은 문구가 들어왔습니다.

"
System.ValueType의 GetHashCode 메서드는 내부적으로 리플렉션과 인스턴스 필드들 중 일부 필드의 값을 XOR 연산을 사용하여 구하도록 되어 있다.
"



위와 같이 언급하기 이전에 GetHashCode에 대한 코드 예제를 하나 제공하고 있는데요. 아래와 같습니다.

internal sealed class Point
{
    private Int32 m_x, m_y;
    public override Int32 GetHashCode()
    {
        return m_x ^ m_y;    
    }
    ...
}

위의 코드는 "참조" 타입이라는 점이 다르고, 또한 CLR에서 System.ValueType의 GetHashCode를 위와 같이 구현되어 있다는 식의 이야기는 없었습니다. 그런데... 여기서 문득 호기심이 생기더군요. ^^ 정말 CLR에서도 위와 같이 구현했을까?




간단한 테스트를 해보면 알 수 있는 문제였습니다. 위에서 제시된 Point.GetHashCode의 구현을 보면 한가지 오류를 직관적으로 발견할 수 있습니다. 즉, m_x와 m_y 값이 서로 바뀐 상태라고 해도 같은 해시 코드값이 나온다는 점입니다. 어떤 구현에서는 값이 바뀌어도 크게 문제가 없을 수도 있지만 분명 문제가 되는 경우도 많습니다. 예를 들자면,

struct GeoLocation
{
    public int Latitude; // 편의상 int로 정의
    public int Longitude;
}

명확하지요. ^^ 지리 좌표가 앞 뒤 값이 바뀌는 것은 용서가 안되는 경우입니다.

자, 그럼 CLR이 정말 단순히 값형식의 경우에 필드값을 XOR해서 구한 것일까요?

GeoLocation t = new GeoLocation();
t.Latitude = 5;
t.Longitude = 6;

int hash1 = t.GetHashCode();  // 결괏값: 373079212 (CLR 2.0 기준)
int hash2 = (int)t.Latitude ^ (int)t.Longitude; // 결괏값: 3

아하... 꼭 그렇지는 않군요. 어떤 다른 변수가 있는 것 같습니다. 일단 그것까지는 관심이 없고.

이제 필드의 값 순서를 바꿔서 해시를 구해볼까요?

GeoLocation t1 = new GeoLocation();
t1.Latitude = 5;
t1.Longitude = 6;

GeoLocation t2 = new GeoLocation();
t2.Latitude = 6;
t2.Longitude = 5;

int hash1 = t1.GetHashCode();  // 결괏값: 373079212
int hash2 = t2.GetHashCode();  // 결괏값: 373079212

이런 ... ^^ 값이 같습니다.

여기서 갑자기 의문이 생겼습니다. GetHashCode를 직접적으로 사용하는 Hashtable과 같은 타입은 키가 같은 데이터가 2번 Add되지 못하도록 합니다. 실제로 아래와 같은 코드는 System.ArgumentException 예외가 발생합니다.

var dict = new Hashtable();
dict.Add(t1, 1);
dict.Add(t1, 2); // System.ArgumentException 예외 발생

그렇다면 다음의 구문도 예외가 발생할까요?

var dict = new Hashtable();
dict.Add(t1, 1);  // t1.GetHashCode == 373079212
dict.Add(t2, 2);  // t2.GetHashCode == 373079212

발생하지 않습니다. 이유는? Hashtable.Insert 메서드를 검사해 보니 GetHashCode 값이 같은 경우에 한해서 이전에 있던 데이터와 새로운 데이터의 Equals 메서드를 호출해서 값이 같은지를 비교하는 코드가 더 추가되어 있었습니다. 즉, t1과 t2가 같지 않으니 해시값은 같지만 인정하고 포함을 시켜주는 것입니다.

확인을 위해 GeoLocation의 Equals 메서드를 다음과 같이 재정의하면, 여지없이 예외가 발생합니다.

struct GeoLocation
{
    public override bool Equals(object obj)
    {
        return obj.GetHashCode() == this.GetHashCode();
    }
}

var dict = new Hashtable();
dict.Add(t1, 1);  // t1.GetHashCode == 373079212
dict.Add(t2, 2);  // t2.GetHashCode == 373079212 // System.ArgumentException 예외 발생




그럼, 제목에서 제시한 이유가 명확해 졌습니다. 위와 같은 이유 때문에, GetHashCode를 Dictionary 타입에 전달할 키로 사용해서는 안되는 것입니다. 사실, 그동안 무심코 써왔던 GetHashCode인데... 이제는 좀 상황을 가려가면서 써야 겠습니다. ^^

그건 그렇고.

그렇다면, 위와 같은 경우에 유일성을 보장하기 위해서 어떻게 GetHashCode를 재정의하는 것이 좋을까요?

애석하게도 이 질문은 잘못되었습니다. "유일성을 보장"하는 것은 욕심이고, "가능한 유일성을 보장"하는 것이 맞습니다. 다들 아시는 것처럼 다양한 정보를 가진 개체가 단지 4byte의 정수값으로 유일성을 보장받는다는 것은 애당초 불가능합니다. 단적으로 아무리 이상적으로 Hash 값을 구하는 경우라 하더라도 UInt32.MaxValue를 넘게 포함하는 데이터에는 무조건 중복이 될 수밖에 없습니다.

즉, 충돌을 감안해야 하고 대신에 이를 해결하기 위해 Equals만 제대로 정의해주면 Hashtable 등의 Dictionary 타입을 사용하는 데에는 (충돌된 해시값을 가진 데이터 검색 시간으로 속도저하가 될 지언정) 문제가 되지 않습니다. 나아가서, 여러분들이 임의로 Hashtable 등의 타입을 만들어준다면 해당 데이터의 GetHashCode만을 사용해서는 안되고 반드시 Equals 검증을 거쳐서 해시값 충돌을 해결해 줘야 합니다.

그래도, 가능하면 충돌을 적게 하는 것이 좋겠지요. ^^
이것은 GetHashCode 계산을 해야하는 값의 업무 도메인 영역마다 틀릴 수 있는 문제입니다. 만약 필드의 앞/뒤값이 틀린 것에 대해서 서로 다른 해시값을 사용하고 싶다면 다음과 같은 식으로 필드마다 일정한 SHIFT 연산을 한 후 XOR 연산을 하는 것도 방법일 수 있습니다.

Make Types Hashable with GetHashCode()
; http://www.c-sharpcorner.com/uploadfile/freebookarticles/samspublishing/2010mar24003215am/VersatileTypes/5.aspx

위의 방법도 사실 정답이 아닐 수 있는데, 오히려 비트 연산하기 전에는 충돌이 안 나던 "경우의 수"가 이제는 충돌이 발생할 수 있기 때문입니다. 다시 원점으로 돌아가서, 그냥 충돌은 인정하고 Equals를 기대하는 것이 맞습니다. 물론, 시간적인 여유가 된다면 가능한 적게 충돌이 나도록 적절한 알고리즘을 업무 도메인 영역에서 충분한 테스트를 거친 후 선택할 수 있다면 좋은 정도겠죠!



[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/10/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2010-07-07 05시26분
[땡초] 역시 완소 정보입니다용^^
[guest]

... 136  137  138  139  140  [141]  142  143  144  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1526정성태11/3/201323102디버깅 기술: 57. C# - double 값에 대한 windbg 확인
1525정성태11/2/201329461.NET Framework: 391. C# - EXE/DLL로부터 추출한 이미지/아이콘의 배경색 투명 처리 [8]
1524정성태11/2/201330325기타: 37. 프로그램에 보여지는 리소스(예: 아이콘) 추출하는 방법 [1]
1523정성태11/2/201326677VS.NET IDE: 81. Visual Studio 확장 도구 AttachToW3WP - w3wp.exe에 대한 디버거 연결을 자동화하는 도구 [2]
1522정성태11/1/201323279VS.NET IDE: 80. IIS 8.0/8.5 - Global.asax.cs처럼 초기에 실행되는 코드에 Breakpoint를 잡는 방법
1521정성태11/1/201329171VS.NET IDE: 79. IIS 7.5 - Global.asax.cs처럼 초기에 실행되는 코드에 Breakpoint를 잡는 방법
1520정성태10/31/201323635오류 유형: 191. Visual Studio 2010 - 웹 애플리케이션 생성 시 "The project type is not supported by this installation." 오류 발생 해결
1519정성태10/31/201349084기타: 36. SYSTEM 또는 TrustedInstaller 소유로 되어 있는 폴더/파일을 삭제하는 방법 [5]
1518정성태10/30/201326747VS.NET IDE: 78. Visual Studio 확장으로 XmlCodeGenerator 제작하는 방법
1517정성태10/28/201326298디버깅 기술: 56. 덤프 파일에 핸들/스레드 정보를 포함하는 방법 [1]
1516정성태10/28/201331663.NET Framework: 390. FolderBrowserDialog보다 더 쓸만한 대화창이 필요하다면? [1]
1515정성태10/24/201334296VS.NET IDE: 77. Visual Studio 확장(VSIX) 만드는 방법 [5]
1514정성태10/24/201367669개발 환경 구성: 202. Internet Explorer 11을 7, 8, 9, 10 버전으로 인식시키는 방법 [9]파일 다운로드1
1513정성태10/23/201324179개발 환경 구성: 201. Azure Blob Storage의 DNS 경로를 사용자 DNS로 바꾸는 방법 [1]
1512정성태10/18/201327387개발 환경 구성: 200. IIS AppPool의 실행 계정을 변경하는 방법
1511정성태10/12/201325516.NET Framework: 389. The 3n + 1 problem의 C#/Java 버전 풀이 [2]
1510정성태10/8/201326418오류 유형: 190. 윈도우 서버 2012 R2 설치 후 인텔 NIC으로 인한 WMI 오류 발생
1509정성태10/8/201331584오류 유형: 189. Windows Server 8.1/2012 R2 - IME 비정상 종료 현상 [1]
1508정성태10/4/201326715.NET Framework: 388. 일반 닷넷 프로젝트에서 WinRT API를 호출하는 방법 [2]파일 다운로드1
1507정성태9/30/201324518오류 유형: 188. The key 'LocalizedPerfCounter' does not exist in the appSettings configuration section.
1506정성태9/30/201326657오류 유형: 187. Parameter "basePath" cannot be a relative path
1505정성태9/26/201375206기타: 35. Microsoft Office 2007 인증 생략하는 방법 [10]
1504정성태9/24/201330110.NET Framework: 387. UDP 브로드캐스팅을 이용해 서비스 측의 IP 주소를 구하는 방법 [1]파일 다운로드1
1503정성태9/21/201335275개발 환경 구성: 199. Visual Studio - github 연동 [7]
1502정성태9/21/201338882개발 환경 구성: 198. Visual Studio - git을 이용한 로컬 소스 컨트롤
1501정성태9/21/201345963개발 환경 구성: 197. Visual Studio를 위한 Git 환경 설정 [5]
... 136  137  138  139  140  [141]  142  143  144  145  146  147  148  149  150  ...