三门问题

三门问题

[TOC]

问题介绍

起源

三门问题(Monty Hall problem)亦称为蒙提霍尔问题、蒙特霍问题或蒙提霍尔悖论,大致出自美国的电视游戏节目Let's Make a Deal。问题名字来自该节目的主持人蒙提·霍尔(Monty Hall)。

描述

参赛者会看见三扇关闭了的门,其中一扇的后面有一辆汽车,另外两扇门后面则各藏有一只山羊。选中后面有车的那扇门可赢得该汽车。当参赛者选定了一扇门,但未去开启它的时候,节目主持人开启剩下两扇门中的一扇,露出其中的山羊。随后,主持人会问参赛者要不要改选另一扇仍然关上的门。

问题

改变选择是否会增加参赛者赢得汽车的机率?

错误的理解

这个问题之所以迷人,是因为无论回答会或者不会的人,往往都给出了错误的解释,下面列出了两种常见的错误解释。

应该改选

理由是,

第一次三选一,主持人排除一扇门后变为二选一,所以应该改选。

这种观点也就认为,改选后选中的概率为\frac{1}{2}.

不应该改选

理由是,

在主持人排除一扇门后,情况已经变成二选一,此时无论是否改选,选中的概率都自动提升为\frac{1}{2}.

这种观点认为,无论是否改选,选中的概率都为\frac{1}{2}.

实验

两种观点似乎都有道理,不如用计算机仿真程序实验一下。

问题建模

仿真程序的流程大致是先利用随机数生成器模拟参赛选手的选择过程,记录仿真结果,再统计中奖频率来估计概率。

仿真过程的结构是一个大循环,重复模拟结果、记录结果。考虑到我们需要比较不同参赛策略的胜率,应该将建模游戏规则(环境)和参赛选手(策略)的代码解耦合,即分别对环境和智能体建模。

环境

对环境(主持人和门)建模:

class Evn:
    
    def __init__(self, doors=['🚗', '🐏', '🐏']):
        self.doors = list(doors)
    
    def is_get_car(self, choice):
        return self.doors[choice] == '🚗'
    
    def give_suggest(self, choice):
        door_range = list(range(len(self.doors)))
        door_range.remove(self.doors.index('🚗'))
        if choice in door_range:
            door_range.remove(choice)
        return random.choice(door_range)

环境向智能体提供两种服务:

  • 为选择给出结果(是否中奖)
  • 为选择给出建议(排除某扇门)

选手

对选手建模:

class Contestant:
    
    def __init__(self, strategy, door_number=3):
        self.door_number = door_number
        self.strategy = strategy
        self.last_choice = 0
    
    def make_choice(self, suggest=None):
        if suggest is not None:
            return self.strategy(suggest, self.last_choice, self.door_number)
        else:
            self.last_choice = random.randrange(self.door_number)
            return self.last_choice

选手根据环境中门的数量和所选策略,有如下行为:

  • 做出初始选择
  • 根据建议再次选择

注意,上文给出的代码把这两种行为集成在make_choice方法中了。

策略

所谓策略,就是根据建议、上次的选择、门的数量(选择范围)做出这次的选择。

  1. 听从建议改变选择的策略

    def strategy_change(suggest, last_choice, door_number):
        doors = list(range(door_number))
        doors.remove(suggest)
        doors.remove(last_choice)
        return random.choice(doors)
    
  2. 不改变选择

    def strategy_not_change(suggest, last_choice, door_number):
        return last_choice
    
  3. 参考建议,在剩下的选项中随机选择

    def strategy_random_change(suggest, last_choice, door_number):
        doors = list(range(door_number))
        doors.remove(suggest)
        return random.choice(doors)
    

随后的流程就是要对比这三种策略的胜率。

流程

对每种策略多次实验,记录结果:

iter_number = 100000
strategies = [strategy_change, strategy_not_change, strategy_random_change]
evn = Evn()
records = {}

for strategy in strategies:
    for i in range(iter_number):
        contestant = Contestant(strategy)
        contestant.make_choice()
        suggest = evn.give_suggest(contestant.last_choice)
        new_choice = contestant.make_choice(suggest)
        if evn.is_get_car(new_choice):
            records[strategy] = records.setdefault(strategy, 0) + 1

# 显示每种方法的胜率
for s in records:
    print(s.__name__, records[s] / iter_number)

每种策略实验十万次,统计中奖比例。

实验结果

策略 胜率 近似于
改变选择 0.66334 \frac{2}{3}
不改变选择 0.33294 \frac{1}{3}
随机改变选择 0.50069 \frac{1}{2}

看来,最初提到的两种解释都是错误的:改选后胜率为\frac{2}{3}而不是\frac{1}{2};不改选胜率保持\frac{1}{3},并不会自动提升。

思考

关于这个结果的标准概率分析容易找到,这里不再赘述。

最困扰我的一点其实是,为什么在主持人排除一个错误选项之后,第一次选项的概率没有提升(从实验结果上看,被排除选项的概率一点也没有加在选手初次的选择上,全部加在了另一个选项上)?毕竟,在获知一个选项被排除后,问题就变成从两扇门里选一个有车的了。

破局关键

关键就在于,主持人给出的建议和选手初次选择是有关系的

这点可以从代码里清楚地看到,Evn类的give_suggest方法有choice这个参数。换句话说,主持人排除的这个选项不是从所有不中奖的门里选的,而是更进一步,不包括选手第一次选择的门。

信息是不确定性的消除。主持人开门是信息,得到这个信息后,每扇门后有车的概率可能有所改变(具体计算参考这篇用贝叶斯公式分析三门问题)。而在这个问题中,由于主持人排除方法的特殊性(在剩下的门中排除),导致信息增量全部落在了另一扇门上。

附录

全部实验代码:

import random


class Evn:
    
    def __init__(self, doors=['🚗', '🐏', '🐏']):
        self.doors = list(doors)
    
    def is_get_car(self, choice):
        return self.doors[choice] == '🚗'
    
    def give_suggest(self, choice):
        door_range = list(range(len(self.doors)))
        door_range.remove(self.doors.index('🚗'))
        if choice in door_range:
            door_range.remove(choice)
        return random.choice(door_range)


class Contestant:
    
    def __init__(self, strategy, door_number=3):
        self.door_number = door_number
        self.strategy = strategy
        self.last_choice = 0
    
    def make_choice(self, suggest=None):
        if suggest is not None:
            return self.strategy(suggest, self.last_choice, self.door_number)
        else:
            self.last_choice = random.randrange(self.door_number)
            return self.last_choice


def strategy_change(suggest, last_choice, door_number):
    doors = list(range(door_number))
    doors.remove(suggest)
    doors.remove(last_choice)
    return random.choice(doors)


def strategy_not_change(suggest, last_choice, door_number):
    return last_choice


def strategy_random_change(suggest, last_choice, door_number):
    doors = list(range(door_number))
    doors.remove(suggest)
    return random.choice(doors)


iter_number = 100000
strategies = [strategy_change, strategy_not_change, strategy_random_change]
evn = Evn()
records = {}
for strategy in strategies:
    for i in range(iter_number):
        contestant = Contestant(strategy)
        contestant.make_choice()
        suggest = evn.give_suggest(contestant.last_choice)
        new_choice = contestant.make_choice(suggest)
        if evn.is_get_car(new_choice):
            records[strategy] = records.setdefault(strategy, 0) + 1
for s in records:
    print(s.__name__, records[s] / iter_number)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容