什么是枪手博弈:

  枪手博弈指彼此痛恨的甲乙丙三个枪手准备决斗。甲枪法最好,十发八中。乙枪法次之,十发六中。丙枪法最差,十发四中。假设他们了解彼此实力,也能做出理性判断。

  问题一:如果三人同时开枪,并且每人每次只发一枪。若干轮枪战后,谁活下来的机会大?

  问题二:如果三人轮流开枪,并且由枪法最差的丙先开枪,他该怎么做?

问题一:

分析:

  3个人同时开枪,则枪手开枪的同时也可能被射死,4种情况,场上分别剩余(3,2,1,0)人

1)场上剩余3人,此时a(甲)一定向b(乙)射击,b一定向a射击,c(丙)可能向a射击或向空射击

2)场上剩余2人,互射,无其他可能

3)场上剩余1人,最后的胜利者

4)场上剩余0人,全部死亡

抽象出枪手模型

 public class Hackbuteer {
private int state=1; //存活状态,1存活,0死亡
private double shot_rate; //命中率
public Hackbuteer(double shot_rate){
this.shot_rate=shot_rate;
}
public int getState() {
return state;
}
public void setState(int state) {
this.state = state;
}
public double getShot_rate() {
return shot_rate;
}
public void setShot_rate(double shot_rate) {
this.shot_rate = shot_rate;
}
public boolean shot(Hackbuteer target) {
double rand = Math.random();
if(rand<shot_rate){ //随机数小于命中率,表示命中
target.state=0; //目标死亡
return true;
}
return false;
}
}

模拟多次(100w)射击,用频率估算概率,代码如下:

 public class HackbuteerTest {
public static void main(String[] args) {
int a_count=0; //a存活次数
int b_count=0; //b存活次数
int c_count=0; //c存活次数
for(int i=0;i<1000000;i++){
Hackbuteer a=new Hackbuteer(0.8);
Hackbuteer b=new Hackbuteer(0.6);
Hackbuteer c=new Hackbuteer(0.4);
while(true){
int all_state=a.getState()+b.getState()+c.getState(); //场上存活状态
if(all_state==3){ //3人都存活
a.shot(b);
b.shot(a);
//c.shot(a); //c向a射击;若模拟向空射击,则注释此行
}else if(all_state==2){ //1人死亡,2人存活
if(a.getState()==0){
b.shot(c);
c.shot(b);
}
if(b.getState()==0){
a.shot(c);
c.shot(a);
}
if(c.getState()==0){
b.shot(a);
a.shot(b);
}
}else if(all_state==1 //仅有一人存活
|| all_state==0 //无人存活
){
if(a.getState()==1){
a_count++;
}
if(b.getState()==1){
b_count++;
}
if(c.getState()==1){
c_count++;
}
break;
} }
}
System.out.println("a存活--"+a_count);
System.out.println("b存活--"+b_count);
System.out.println("c存活--"+c_count);
System.out.println("无人存活--"+(1000000-a_count-b_count-c_count)); }
}

结果:

c向a射击,c向空射击

所以对于c来说,在同时开枪的情况下,c应该向a开枪,而不是向空开枪。

PS:以上结果仅代表实验100w次的频率,不完全代表概率。

问题二:

分析:

3人轮流开枪,c先开枪,依然是2种方案(向a射击,向空射击),代码如下:

 public class HackbuteerTest2 {
public static void main(String[] args) {
int a_count=0; //a存活次数
int b_count=0; //b存活次数
int c_count=0; //c存活次数
for(int i=0;i<1000000;i++){
Hackbuteer a=new Hackbuteer(0.8);
Hackbuteer b=new Hackbuteer(0.6);
Hackbuteer c=new Hackbuteer(0.4);
while(true){
//c存活,开枪策略1,优先射击a
if(c.getState()==1){
if(a.getState()==1){
c.shot(a);
}else if(b.getState()==1){
c.shot(b);
}
}
//c存活,开枪策略2,当a,b都存活时,不射击任何人
// if(c.getState()==1){
// if(a.getState()==0 && b.getState()==1){
// c.shot(b);
// }else if(a.getState()==1 && b.getState()==0){
// c.shot(a);
// }
// } //a存活,开枪策略,优先射击b
if(a.getState()==1){
if(b.getState()==1){
a.shot(b);
}else if(c.getState()==1){
a.shot(c);
}
}
//b存活,开枪策略,优先射击a
if(b.getState()==1){
if(a.getState()==1){
b.shot(a);
}else if(c.getState()==1){
b.shot(c);
}
} if(a.getState()+b.getState()+c.getState()==1){ //仅有一人存活
if(a.getState()==1){
a_count++;
}
if(b.getState()==1){
b_count++;
}
if(c.getState()==1){
c_count++;
}
break;
} }
}
System.out.println("a存活--"+a_count);
System.out.println("b存活--"+b_count);
System.out.println("c存活--"+c_count);
}
}

结果:

c向a射击,c向空射击

显然,此时c应该向空射击,而不是向a射击

PS:当射击顺序为(c-b-a时,结论依然如此)

结论:

火并也要按照基本法,同时开枪有1/4-1/3的概率全部玩完(你是不是以为我要说什么实力很重要,选择很重要,背叛与忠诚?)

[java,2019-01-28] 枪手博弈,谁才是最后赢家的更多相关文章

  1. [Java 教程 01] Hello,Java!

    前言 从事编程已经有一段时间了,突然发现,Java作为我的第一编程语言,自己似乎对她并有一个系统的思想.当下Java依旧保持着超高的热度,新特性也不断出现,从当初学习的java6版本到最近刚出的jav ...

  2. Tensorflow学习笔记2019.01.22

    tensorflow学习笔记2 edit by Strangewx 2019.01.04 4.1 机器学习基础 4.1.1 一般结构: 初始化模型参数:通常随机赋值,简单模型赋值0 训练数据:一般打乱 ...

  3. 091 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 02 static关键字 01 static关键字(上)

    091 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 ...

  4. Java学习01

    Java学习01 第一章 1.JRE与JDK JDK(JAVA Develop Kit,JAVA开发工具包)提供了Java的开发环境和运行环境,主要用于开发JAVA程序,面向Java程序的开发者; J ...

  5. 「FFT」题单(upd 2019.4.28)

    持续更新(last upd 2019.4.28) ZJOI2014 力 [题目链接] 解法 对原式进行转换,然后卷积FFT套上去求解就可以了. 推导过程简洁版: \[F_i=\sum_{j<i} ...

  6. Tensorflow学习笔记2019.01.03

    tensorflow学习笔记: 3.2 Tensorflow中定义数据流图 张量知识矩阵的一个超集. 超集:如果一个集合S2中的每一个元素都在集合S1中,且集合S1中可能包含S2中没有的元素,则集合S ...

  7. AWS re:Invent(2019.01.09)

    时间:2019.01.09地点:北京国际饭店

  8. Alpha冲刺(5/10)——2019.4.28

    所属课程 软件工程1916|W(福州大学) 作业要求 Alpha冲刺(5/10)--2019.4.28 团队名称 待就业六人组 1.团队信息 团队名称:待就业六人组 团队描述:同舟共济扬帆起,乘风破浪 ...

  9. 2019.3.28&2019.3.30考试

    2019.3.28 : 肥肠爆芡,因为这场考试的题太屑了,所以我咕咕了 Upd on 2019.3.30 压进来一篇(因为都没啥意义) 2019.3.30 : 全机房读错题+没有大样例=T2全体爆炸 ...

随机推荐

  1. onvif 框架代码生成

    1:gsoap官网(http://gsoap2.sourceforge.net/)下载最新版gsoap(本次版本为gsoap_2.8.17)并解压. 2:新建一个文件夹(OnvifFramework) ...

  2. Axure RP 介绍

    原型设计是将想法转变为设计过程中至关重要的一环.经常有设计师小伙伴可能会问到,“哪个原型设计工具是最好的呢”?实际上这是一种错误的提问方式,尤其是在当下原型工具种类繁多,针对不同需求各有优势的大环境中 ...

  3. ubuntu18安装微信

    1.从git下载tar包 wget http://github.com/geeeeeeeeek/electronic-wechat/releases/download/V2.0/linux-x64.t ...

  4. 渗透测试学习 三、Linux基础

    Linux发行版本  内核+应用程序  打包在一起 一.优点: 完全免费 完全兼容POSIX 1.0标准 多用户,多任务 良好的界面 可靠安全稳定的性能 支持多种平台 丰富的网络功能 安全性更好(针对 ...

  5. Mac os x 系统的发展史

    ·Mac OS 9:发布时间:1999年 于1999年发布的Mac OS 9操作系统(图片来自互联网) 在OS X之前,1999年发布的Mac OS 9看起来就是一个普通的桌面操作系统.并且现在已经被 ...

  6. Scala基础

    1.seq[String] 用[]表示参数类型,java使用<>表示参数,因为Scala 允许以<命名方法和变量名,java不允许 2.支持range的类型:Char .Int . ...

  7. 决策树(Decision Tree

    转化自:https://trainings.analyticsvidhya.com/courses/course-v1:AnalyticsVidhya+LPDS2019+LPDS2019_T1/cou ...

  8. ubutu16.04 安装Tenda u12无线网卡驱动

    ubutu16.04 安装Tenda u12无线网卡驱动 一些问题: 1) Tenda u12 linux版本的驱动支持 kernel 2.6 到 4.4,而前系统内版本核为4.10,所以编译不过去啦 ...

  9. NodeJS 学习笔记

    1. NodeJs的事件模型被称为非阻塞式IO或者事件驱动IO 2. Node.js 几乎每一个 API 都是支持回调函数的. 3. Node.js 基本上所有的事件机制都是用设计模式中观察者模式实现 ...

  10. 为什么使用消息队列? 消息队列有什么优点和缺点? Kafka、ActiveMQ、RabbitMQ、RocketMQ 都有什么区别,以及适合哪些场景?

    https://blog.csdn.net/Iperishing/article/details/86674084