C#实现的海盗分金算法实例
作者:普若伽门 发布时间:2023-12-20 21:00:53
本文实例讲述了C#实现的海盗分金算法。分享给大家供大家参考,具体如下:
海盗分金的故事
5个海盗抢到了100颗宝石,每一颗都一样的大小和价值连城。
他们决定这么分:
1。抽签决定自己的号码(1,2,3,4,5)
2。首先,由1号提出分配方案,然后大家5人进行表决,当且仅当半数和超过半数的人同意时,按照他的提案进行分配,否则将被扔入大海喂鲨鱼。
3。如果1号死后,再由2号提出分配方案,然后大家4人进行表决,当且仅当半数和超过半数的人同意时,按照他的提案进行分配,否则将被扔入大海喂鲨鱼。
4。依次类推......
问题:第一个海盗提出怎样的分配方案才能够使自己的收益最大化
条件:每个海盗都是很聪明的人,如果前面的人提出的方案对自己没好处肯定会否决,如果好处比后面持续下去的方案好就投票。
解决:网上很多解决方法(百度百科:http://baike.baidu.com/view/5221.htm ),下面就是算法总结,目的就是让自己得到1半或以上的票。
算法:从后向前来推理,
i 海盗分为1-5号,如果只剩下第4,5号海盗两个人分配,4号则给自己投一票>=50%,条件成立,自己独吞总金币,5号什么也得不到。
ii 3号推出了4号的方案,发一枚金币给5号,拉一票,因为5号知道在4号的方案中自己得不到所以投3号一票,加上3号投自己的一票>=50%条件成立,3号获得100-1=99枚金币。
iii 2号得出3号方案,给4号一枚金币拉一票,同理,2号票数(1+1)/4>=50%条件成立,获得100-1=99枚金币。
iv 1号推断2号方案中,3号和5号不能获得金币,于是给他们各一枚金币则拉两票,(1+1+1)/5>=50%条件成立,自己获得100-1-1=98枚金币。
从上面的推论可以看出,从后向前依次推,如果上一次分配中获得金币的海盗本次分配中将不能获得金币。
using System;
class pirateAssignGold
{
public static void Main()
{
int pirates=5; //海盗总数
int gold=100; //金币总数
int joinNum; //加入分配的海盗数
int[] poke=new int[pirates+1]; //每个海盗一个口袋
int ticket; //票数计数器
for(int i=pirates;i>=1;i--){
joinNum=pirates-i+1; //此次加入分配的海盗数
ticket=0;
for(int j=pirates;j>=i;j--)
{
if((pirates-j+1)==joinNum) //如果本海盗就是此次加入分配的最后一个海盗
{
poke[j]=gold; //利益最大化,把还剩的金币全给他
gold=gold-poke[j];
ticket=ticket+1;
}
else
{
if(poke[j]>0) //此海盗已经获得了金币
{
gold=gold+poke[j]; //推论中本次分配者会使上一次获得金币的海盗什么都没有。
poke[j]=0;
}
else
{
poke[j]=1; //推论中上一次分配中没有获得金币的海盗会在本次获得金币。
gold=gold-1;
ticket=ticket+1;
}
}
}
if((double)ticket/(double)joinNum<0.5){ break;} //总得票数/此次加入分配的海盗数>=50%则此次分配成立,否则失败
}
for(int n=1;n<=5;n++){
Console.WriteLine("海盗{0}获得金币数{1} ",n,poke[n]);
}
Console.ReadKey();
}
}
希望本文所述对大家C#程序设计有所帮助。


猜你喜欢
- kotlin基础教程之类和继承类声明使用class关键字声明类,查看其声明格式:: modifiers ("class"
- 实体类时间格式化java 实体类 时间格式化注解@JsonFormat(pattern = "yyyy-MM-dd HH:mm:s
- 一、注解的概念1、注解官方解释注解叫元数据,一种代码级别的说明,它是JDK1.5及以后版本引入的一个特性,与类、接口、枚举在同一个层次,它可
- 本文采用C#实例讲解了处理图片为浮雕效果的实现方法,这在PS中是一个常见的功能,也是C#中的一个简单的图像处理例子。程序先读取原图,然后依次
- 前言String可以说是Java中使用最多最频繁、最特殊的类,因为同时也是字面常量,而字面常量包括基本类型、String类型、空类型。一.
- 面试题1:你了解线程池么?简单介绍一下。java提供的一个java.util.concurrent.Executor接口的实现用于创建线程池
- 本文实例讲述了Android开发之DialogFragment用法。分享给大家供大家参考,具体如下:背景Android 官方推荐使用 Dia
- try catch finally组合:检测异常,并传递给catch处理,并在finally中进行资源释放。try catch组合 : 对代
- 日常开发中,判断邮箱是少不了的,这个我以**C#**为例,来写一个判断方法,正则表达式是通用的,CV就可以首先引入正则需要使用的命名空间//
- 在使用spring框架中我们都知道,某个类如果使用了@Service、@Autowire 这种依赖注入的方式引用了其他对象,在另外一个类中,
- 最近几年玩得最疯狂的应该是发红包了,尤其是过年的时候特别受欢迎,下面写了红包的随机算法,其实挺简单的,仅是提供一种思路,希望可以给大家一些启
- 该方法把该字符串转换成一个新的字符数组。 String str="abcdefg"; char a[]; a=str.t
- 本文实例讲述了C#中委托用法。分享给大家供大家参考。具体分析如下:这里演示了如何使用匿名委托来计算员工的薪水奖金。使用匿名委托简化了程序,因
- 1、正常dubbo调用流程引入dubbo依赖引入他人提供的clinet依赖包;配置相同的注册中心,使用@Reference注解注入对应的se
- 本文实例讲述了简单记事本java实现代码。分享给大家供大家参考。具体如下:完整代码如下:import java.awt.*;import j
- 1、特效按钮的进展 之前的思路:css设置div的样式,在js中实现div对事件的响应,并改变div的样式,以实现动画效果。 1:以动画的形
- spring data JPA的多属性排序在此介绍我所用的一种方式:第一步,引包import org.springframework.dat
- 1. 简单工厂介绍简单工厂有一个具体的工厂类,可以生产不同的产品,属于创建型设计模式。注意:简单工厂模式 不属于23种设计模式之列2. 简单
- 一. 线程池简介1. 线程池的概念:线程池就是首先创建一些线程,它们的集合称为线程池。使用线程池可以很好地提高性能,线程池在系统启动时即创建
- 本文将以C#程序代码为例介绍如何来读取txt文件中的内容,生成Word文档。在编辑代码前,可参考如下代码环境进行配置:Visual Stud