Microsoft MVP성태의 닷넷 이야기
Math: 11. C# 시뮬레이션 - 몬티홀 게임 [링크 복사], [링크+제목 복사],
조회: 26559
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 31  32  33  34  [35]  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13098정성태7/13/202215894개발 환경 구성: 647. Azure - scale-out 상태의 App Service에서 특정 인스턴스에 요청을 보내는 방법 [1]
13097정성태7/12/202214876오류 유형: 817. Golang - binary.Read: invalid type int32
13096정성태7/8/202218116.NET Framework: 2030. C# 11 - UTF-8 문자열 리터럴
13095정성태7/7/202215576Windows: 208. AD 도메인에 참여하지 않은 컴퓨터에서 Kerberos 인증을 사용하는 방법
13094정성태7/6/202215189오류 유형: 816. Golang - "short write" 오류 원인
13093정성태7/5/202215772.NET Framework: 2029. C# - HttpWebRequest로 localhost 접속 시 2초 이상 지연
13092정성태7/3/202217854.NET Framework: 2028. C# - HttpWebRequest의 POST 동작 방식파일 다운로드1
13091정성태7/3/202216405.NET Framework: 2027. C# - IPv4, IPv6를 모두 지원하는 서버 소켓 생성 방법
13090정성태6/29/202216022오류 유형: 815. PyPI에 업로드한 패키지가 반영이 안 되는 경우
13089정성태6/28/202216715개발 환경 구성: 646. HOSTS 파일 변경 시 Edge 브라우저에 반영하는 방법
13088정성태6/27/202215015개발 환경 구성: 645. "Developer Command Prompt for VS 2022" 명령행 환경의 폰트를 바꾸는 방법
13087정성태6/23/202218435스크립트: 41. 파이썬 - FastAPI / uvicorn 호스팅 환경에서 asyncio 사용하는 방법 [1]
13086정성태6/22/202218330.NET Framework: 2026. C# 11 - 문자열 보간 개선 2가지파일 다운로드1
13085정성태6/22/202217561.NET Framework: 2025. C# 11 - 원시 문자열 리터럴(raw string literals)파일 다운로드1
13084정성태6/21/202217071개발 환경 구성: 644. Windows - 파이썬 2.7을 msi 설치 없이 구성하는 방법
13083정성태6/20/202217463.NET Framework: 2024. .NET 7에 도입된 GC의 메모리 해제에 대한 segment와 region의 차이점 [2]
13082정성태6/19/202216432.NET Framework: 2023. C# - Process의 I/O 사용량을 보여주는 GetProcessIoCounters Win32 API파일 다운로드1
13081정성태6/17/202215109.NET Framework: 2022. C# - .NET 7 Preview 5 신규 기능 - System.IO.Stream ReadExactly / ReadAtLeast파일 다운로드1
13080정성태6/17/202216704개발 환경 구성: 643. Visual Studio 2022 17.2 버전에서 C# 11 또는 .NET 7.0 preview 적용
13079정성태6/17/202213941오류 유형: 814. 파이썬 - Error: The file/path provided (...) does not appear to exist
13078정성태6/16/202217489.NET Framework: 2021. WPF - UI Thread와 Render Thread파일 다운로드1
13077정성태6/15/202218533스크립트: 40. 파이썬 - PostgreSQL 환경 구성
13075정성태6/15/202215209Linux: 50. Linux - apt와 apt-get의 차이 [2]
13074정성태6/13/202216044.NET Framework: 2020. C# - NTFS 파일에 사용자 정의 속성값 추가하는 방법파일 다운로드1
13073정성태6/12/202216470Windows: 207. Windows Server 2022에 도입된 WSL 2
13072정성태6/10/202216551Linux: 49. Linux - ls 명령어로 출력되는 디렉터리 색상 변경 방법
... 31  32  33  34  [35]  36  37  38  39  40  41  42  43  44  45  ...