Microsoft MVP성태의 닷넷 이야기
Math: 11. C# 시뮬레이션 - 몬티홀 게임 [링크 복사], [링크+제목 복사],
조회: 26593
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

C# 시뮬레이션 - 몬티홀 게임

회사에 ^^ 이런 재미있는 소재를 이야기 해주는 직원이 있습니다. 어느 날은 몬티홀 게임 문제를 냈는데요. 초보 통계 책 한번이라도 보신 분들은 이와 비슷한 유형의 문제를 본 적이 있기 때문에 쉽게 풀(찍을) 수 있습니다.

일단, 기본적인 상황은 다음과 같습니다.

  1. 3개의 카드가 있고, 그 중 하나의 카드에만 페라리가 그려져 있습니다.
  2. 섞여진 카드에서 여러분은 카드 하나를 펼치고, 그것이 페라리라면 여러분이 갖는 것입니다.

자, 위의 문제만 보면 여러분은 1/3의 확률로 페라리를 가질 수 있다는 것은 누구나 알 수 있습니다.

그런데 몬티홀 게임에서는 다음과 같은 가정이 하나 들어갑니다.

  1. 3개의 카드가 있고, 그 중 하나의 카드에만 페라리가 그려져 있습니다.
  2. 섞여진 카드에서 여러분은 카드 하나를 (펼치지 않고) 선택합니다.
  3. 사회자가 선택되지 않은 나머지 2개의 카드 중에서 페라리가 아닌 그림을 펼쳐주고 여러분들에게 다시 한번 선택의 기회를 줍니다.

여러분은 이 상황에서 기존에 선택한 것을 취소하고 남아있는 다른 카드를 선택하시겠습니까? 아니면 기존 카드를 고수하시겠습니까? ^^

기존 카드를 고수하면 페라리를 가질 확률은 1/3이지만, 선택을 바꾼다면 페라리를 가질 확률은 2/3로 굉장히 높아집니다.




어떻게 그런 식으로 확률이 변하는지에 대해 이해하려면 간단한 확률 트리를 그려보면 됩니다. 우선, 사회자가 아무런 간섭을 하지 않았을 때는 누가 봐도 1/3의 확률이므로 그건 그려볼 필요도 없겠고. 이제 사회자가 간섭하는 경우를 따져 보겠습니다.

일단, 여러분들이 선택할 경우의 수는 "정답", "꽝1", "꽝2"로 3가지가 있습니다. 그 3가지에 대해 각각 사회자가 간섭했을 때 여러분들이 선택을 바꾼다면 어떻게 되는지 간단하게 따져 보겠습니다.

  • 처음에 "정답"을 선택한 경우, 선택을 바꾼다면 나머지 2개는 모두 "꽝"이므로 무조건 "꽝"이 됨.
  • 처음에 "꽝1"을 선택한 경우, 사회자는 "꽝2"를 펼쳐줄 것이므로 선택을 바꾼다면 무조건 "정답"이 됨.
  • 처음에 "꽝2"를 선택한 경우, 사회자는 "꽝1"을 펼쳐줄 것이므로 선택을 바꾼다면 무조건 "정답"이 됨.

보셨죠? ^^ 사회자가 남아있는 "꽝"을 하나 보여준 경우 여러분들이 선택을 바꾼다면 3번의 시도 중에 2번은 "정답"으로 자연스럽게 선택이 되는 것입니다.

즉, 몬티홀 게임 문제는 따져보면 결국 다음과 같은 하나의 문장으로 표현을 바꿀 수 있습니다.

몬티홀 게임 확률은 3개의 카드 중에 "꽝"을 선택할 확률과 같다.

직원이 소개해 준 다음의 위키에 이 게임에 대한 설명이 아주 자세하게 나와 있습니다.

몬티홀 게임 
; http://rigvedawiki.net/r1/wiki.php/%EB%AA%AC%ED%8B%B0%ED%99%80

몬티홀 게임 - 시뮬레이션
; http://www.grand-illusions.com/simulator/montysim.htm

참고로, 저도 시뮬레이션을 C#으로 만들어 보았습니다.

우선 3개의 카드를 표현한 후 선택을 모두 취소시키고,

this.ComputerSide = new Side[3];
this.UserSide = new Side[3];

for (int i = 0; i < this.ComputerSide.Length; i++)
{
    this.ComputerSide[i] = new Side();
    this.ComputerSide[i].Index = i;

    this.UserSide[i] = new Side();
    this.UserSide[i].Index = i;
}

Array.ForEach(this.ComputerSide, e => e.Clear());
Array.ForEach(this.UserSide, e => e.Clear());

랜덤 함수를 이용해 페라리가 그려진 카드가 위치할 번호를 컴퓨터가 정하게 하고,

// 컴퓨터 측의 정답 선택
int computerChoice = _rand.Next(0, 3);
this.ComputerSide[computerChoice].Current = true;

이어서 사용자 측의 답도 선택하게 합니다.

int userChoice = _rand.Next(0, 3);

만약 이 상태에서 computerChoice와 userChoice를 비교하는 것으로 마무리 지으면 일반적인 1/3의 확률로 사용자가 페라리 카드를 맞추게 됩니다.

하지만 사회자가 사용자가 선택한 카드를 제외한 나머지 2개의 카드를 구하고,

// 사용자가 선택한 것을 제외한 카드를 얻고,
var remains = this.UserSide.Where((e) => e.Index != userChoice);

남은 2장의 카드에서 컴퓨터가 정답으로 선택하지 않은 항목을 찾아 사용자가 선택하면,

// 남은 선택 중에서 컴퓨터가 정답으로 선택하지 않은 항목을 찾아,
Side slotToDelete = remains.Where((e) => e.Index != computerChoice).First();

// 그 항목을 제외한 나머지 카드의 번호를 사용자가 새롭게 선택한 것으로 채택
int newUserChoice = remains.Where((e) => e != slotToDelete).First().Index;

이제 newUserChoice와 최초의 컴퓨터가 선택한 computerChoice의 값을 비교해서 일치 여부를 결정할 수 있고 그 횟수를 세어 확률을 구하면 됩니다.

참고로, 이렇게 해서 시뮬레이션을 돌리니 다음과 같이 66%의 확률로 여러분들이 맞출 수 있다고 나오는 군요. ^^

montysim_1.png

위의 코드 파일은 첨부해 두었습니다. (시뮬레이션에 충실하기 위해 일부러 사용자와 사회자가 할 수 있는 모든 스텝을 코드로 넣었기 때문에 다소 비효율적인 동작을 하게 되었음을 감안해 주세요. ^^)





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

[연관 글]






[최초 등록일: ]
[최종 수정일: 1/9/2014]

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

비밀번호

댓글 작성자
 




... 136  137  138  139  140  141  142  143  144  [145]  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1460정성태6/17/201326483.NET Framework: 372. PerformanceCounter - Category does not exist. [1]
1459정성태6/15/201330102Windows: 74. 한글 키가 아닌 영문 키를 기본으로 선택하는 방법 [5]
1458정성태6/13/201330941.NET Framework: 371. CAS Lock 방식이 과연 성능에 얼마나 도움이 될까요? [1]파일 다운로드1
1457정성태6/13/201326907개발 환경 구성: 192. "Probabilistic Programming and Bayesian Methods for Hackers" 예제 코드 실행 방법
1456정성태6/5/201335553.NET Framework: 370. C# - WebKit .NET 사용 [2]파일 다운로드1
1455정성태6/1/201329525.NET Framework: 369. ThreadPool.QueueUserWorkItem의 실행 지연 [4]파일 다운로드1
1454정성태5/31/201327453Java: 15. Java 7 Control Panel 실행시키는 방법
1453정성태5/22/201326577기타: 32. Microsoft FTP 사이트에 접속하는 방법
1452정성태5/21/201334128Windows: 73. TabProcGrowth 값 삭제 후 IE를 실행시키면 다시 복원되는 경우 [3]
1451정성태5/17/201333032Windows: 72. 윈도우 서버 2012 기초 사용법
1450정성태5/16/201323908오류 유형: 176. SQL10007N Message "0" could not be retrieved. Reason code: "3"
1449정성태5/15/201330964오류 유형: 175. SpeechRecognitionEngine 사용 시 오류 유형 2가지
1448정성태5/14/201325989VC++: 68. #pragma warning(disable: ...)로 오류 제어가 안된다면?
1447정성태5/3/201328044개발 환경 구성: 191. Debugging Tools for Windows 독립 설치 버전 [1]
1446정성태4/30/201328653.NET Framework: 368. Encoding 타입의 대체(fallback) 메카니즘 [1]
1445정성태4/26/201326732디버깅 기술: 54. NT 서비스의 Main 메서드 안에서 Process.GetProcessesByName 호출 시 멈춤 현상 [1]
1444정성태4/26/201330787기타: 31. Internet Explorer: 자바스크립트로 숨겨진 파일 다운로드 경로를 알아내는 방법 [1]
1443정성태4/24/201326031개발 환경 구성: 190. Azure PaaS 웹 응용 프로그램 배포 후 SMTP 서버 구성 [2]
1442정성태4/21/201330146기타: 30. 마이크로소프트 워드의 CPU 점유 현상으로 글자 입력이 느려졌다면? [1]
1441정성태4/21/201336631.NET Framework: 367. LargeAddressAware 옵션이 적용된 닷넷 32비트 프로세스의 가용 메모리 [14]
1440정성태4/19/201325236오류 유형: 174. dumpbin.exe 실행시 mspdb110.dll 로드 오류
1439정성태4/18/201329139VS.NET IDE: 76. Visual Studio 2012와 Itanium 빌드 옵션 [2]
1438정성태4/17/201328778.NET Framework: 366. 다른 프로세스에 환경 변수 설정하는 방법 - 두 번째 이야기 [1]파일 다운로드1
1437정성태4/17/201328947VC++: 67. CRT(C Runtime DLL: msvcr...dll)에 대한 의존성 제거
1436정성태4/17/201333972.NET Framework: 365. Local SYSTEM 권한으로 코드를 실행하는 방법파일 다운로드1
1435정성태4/15/201343205Windows: 71. ad-hoc 보다 더 편리한 "가상 Wifi" 를 이용한 인터넷 공유 [2]
... 136  137  138  139  140  141  142  143  144  [145]  146  147  148  149  150  ...