Python Monty Hall Problem Simulation

The Monty Hall Problem is a very (to me at least) counter-intuitive probability mind-experiment which contorts my brain and fascinates me at the same time, I have been mulling it over the last few weeks and wanted to write a little simulator to see if the numbers come out as predicted (if not expected, and indeed they do!). I can just about understand the probabilistic arguments, but I still find it very confusing, as soon as I think that I grok it – my ‘understanding’ disappears into the night! I am used to being this bamboozled when reading about quantum mechanics or something, but I find it fascinating that such an apparently simple problem can be so deceptively deep!

Summary

There are 3 doors, behind one lies a car, while behind the other two are goats. A player chooses a door at random. Monty opens one of the other doors to show that there is a goat behind it. Monty then asks the player if they would like to stick with their original choice of door or switch to the other un-opened door.

If the player sticks with their door then their chance of winning the car should be 1/3. If the player switches door then their chances of winning the car increases to 2/3!!!

If you are having problems understanding the outcome, I find it helps to imagine that there are a million doors rather than 3. After you choose your door (1/1,000,000 chance of hiding the car) Monty opens up 999,998 doors that hide goats to leave one door still closed. Now which door do you think is most likely to hide the car? The one you choose, or the one that Monty avoided opening while he opened all 999,998 other doors?! It seems obvious to me that the other door that Monty left un-opened has a massively higher chance of hiding the car than your original choice! As N reduces to 3 this ‘obviousness’ reduces greatly however! This simulator allows you to experiment with more then 3 doors for this reason.

For more info: https://en.wikipedia.org/wiki/Monty_Hall_problem

This program is an experimental simulator to see what numbers we get when the player decides to stick or switch. It can be found on GitHub.

#! /usr/bin/python
#   Copyright 2019 Kevin Godden
#
#   Licensed under the Apache License, Version 2.0 (the "License");
#   you may not use this file except in compliance with the License.
#   You may obtain a copy of the License at
#
#       http://www.apache.org/licenses/LICENSE-2.0
#
#   Unless required by applicable law or agreed to in writing, software
#   distributed under the License is distributed on an "AS IS" BASIS,
#   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
#   See the License for the specific language governing permissions and
#   limitations under the License.
import random
"""
    Simulates the Monty Hall problem with 3 doors
    The Monty Hall problem is a very (to me at least) counter-intuitive probability mind experiment which
    contorts my brain and fascinates me at the same time. I can just about understand the
    probabilistic arguments, but I still find it confusing.  This program is an experimental
    simulator to see what numbers we get when the player decides to stick or switch.
    If you are having problems understanding the outcome, I find it helps to imagine that there
    are a million doors rather than 3.  After you choose your door (1/1,000,000 chance of hiding the car)
    Monty opens up 999,998 doors that hide goats to leave one door still closed.  Now which door do you think
    is most likely to hide the car? The one you choose, or the one that Monty avoided opening while
    he opened all 999,998 other doors?!  It seems obvious to me that the other door that Monty left un opened
    has a massively higher chance of hiding the car than the original choice!  As N reduces to 3 this
    obviousness reduces greatly however!  This simulation can be run with more than 3 doors to experiment
    with this...
    Summary:
    --------
    There are 3 doors, behind one lies a car, while behind the other two are goats.
    A player chooses a door at random.  Monty then opens one of the other doors to show that
    there is no car behind it.  Monty asks the player if they would like to stick with their
    door or switch to the other un-opened door.
    If the player sticks with their door then their chance of winning the car should be 1/3
    If the player switches door then their chances of winning the car increases to 2/3!!!
    If you want to run the game with more doors to see how the probabilities change
    just change the door_count value below.
    For more info: https://en.wikipedia.org/wiki/Monty_Hall_problem
"""
# Should the player switch their guess or stick
# with their original one?
switch = True
# The number of doors in the game, should be 3 for
# the problem as normally stated.
door_count = 3
# The number of times to run the simulation
game_count = 10000
# Should we print lots of messages?
# If door_count is large change this to True to
# make simulation run faster.
quiet = False
def pick_random_door():
    """
    Pick a door at 'random', doors are 0-indexed so this will
    return a number between 0 and door_count - 1
    :return: the chosen door
    """
    return random.randint(0, door_count - 1)
def pick_monty_doors(car_door, players_guess):
    """
    Pick the doors for Monty to open, of the number
    of doors is 3 then Monty opens up the first door that doesn't
    have a car and that play hasn't already chosen...
    If door_count > 3 then Monty will open up all of the doors
    that hide goats avoiding the door that has been already picked
    :param car_door: The door behind which the car resides
    :param players_guess: The door which the play choose
    :return: A list of the indices of the doors that Monty opens
    """
    monty_doors = []
    for d in range(0, door_count):
        if len(monty_doors) == door_count - 2:
            break
        if d == car_door:
            continue
        if d == players_guess:
            continue
        monty_doors.append(d)
    return monty_doors
def pick_other_door(players_guess, monty_doors):
    """
    The player has chosen to switch their chosen door, so we
    need to pick another one for them.  We pick the door that
    a.) Doesn't match the player's original guess
    b.) Wasn't chosen by Monty
    :param players_guess:  The door that the player originally picked
    :param monty_doors: The doors that Monty has opened (usually just 1)
    :return: The other door's index
    """
    for d in range(0, door_count):
        # Can't switch to the player's original guess
        if d == players_guess:
            continue
        # Don;t pick a door that Monty has opened
        if d in monty_doors:
            continue
        return d
def run_game():
    """
    Run a single game
    :return:  win/loose --> True/False
    """
    # Pick the door with the car
    car_door = pick_random_door()
    if not quiet:
        print("Car is behind door %d" % car_door)
    # Player chooses a door
    players_guess = pick_random_door()
    if not quiet:
        print("Player has guessed door %d" % players_guess)
    # Monty opens up a door that doesn't have a
    # car behind it.
    monty_doors = pick_monty_doors(car_door, players_guess)
    if not quiet:
        print("Monty opens: " + str(monty_doors))
    # Does the player switch doors after Monty
    # opens his??
    if switch:
        players_guess = pick_other_door(players_guess, monty_doors)
        if not quiet:
            print("Player switches to door %d" % players_guess)
    if not quiet:
        if players_guess == car_door:
            print("Player wins!")
        else:
            print("Player looses ;-(")
    return players_guess == car_door
def run_games():
    wins = 0
    game = 0
    while game < game_count:
        game += 1
        # Run a game and see if we win!
        if run_game():
            wins += 1
    print("Games: %d" % game)
    print("Wins: %d" % wins)
    print("------------------------------")
    if wins == 0:
        print("you didn't win any games!")
    else:
        win_rate = wins / game
        print("Win rate: %f" % win_rate)
if __name__ == "__main__":
    run_games()