C#实现的海盗分金算法实例

 更新时间:2017年07月13日 12:04:55   作者:普若伽门  
这篇文章主要介绍了C#实现的海盗分金算法,结合具体实例形式分析了海盗分金算法的原理与C#相应实现技巧,需要的朋友可以参考下

本文实例讲述了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#相关内容感兴趣的读者可查看本站专题:《C#数据结构与算法教程》、《C#程序设计之线程使用技巧总结》、《C#常见控件用法教程》、《WinForm控件用法总结》、《C#数组操作技巧总结》及《C#面向对象程序设计入门教程

希望本文所述对大家C#程序设计有所帮助。

相关文章

  • C#中字符串的加密的源码

    C#中字符串的加密的源码

    C#中字符串的加密的源码...
    2007-03-03
  • C#实现TCP和UDP通信的示例详解

    C#实现TCP和UDP通信的示例详解

    这篇文章主要为大家详细介绍了C#实现TCP和UDP通信的相关知识,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2023-03-03
  • C#区分中英文按照指定长度截取字符串的方法

    C#区分中英文按照指定长度截取字符串的方法

    这篇文章主要介绍了C#区分中英文按照指定长度截取字符串的方法,涉及C#操作字符串的正则匹配与截取等常用操作技巧,非常具有实用价值,需要的朋友可以参考下
    2015-03-03
  • C#中WPF依赖属性的正确学习方法

    C#中WPF依赖属性的正确学习方法

    这篇文章主要介绍了C#中WPF依赖属性的正确学习方法 ,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-08-08
  • c#基础——了解程序结构

    c#基础——了解程序结构

    这篇文章主要介绍了c# 程序结构的相关资料,文中讲解非常细致,帮助大家更好的理解和学习C#,感兴趣的朋友可以了解下
    2020-07-07
  • C#实现拷贝文件到另一个文件夹下

    C#实现拷贝文件到另一个文件夹下

    这篇文章主要介绍了C#实现拷贝文件到另一个文件夹下,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • c#高效比对大量图片的实例代码

    c#高效比对大量图片的实例代码

    以前传统的比较方式是遍历图片中的每一个像素,然后进行比对。这样的比对在少量图片的比对上虽然效率低一点,但是也没有什么不好。但是在大量图片比对的时候,过长的反应时间和对服务器比较高的消耗肯定是不行的,下面介绍下新的方法
    2013-10-10
  • 在winform下实现左右布局多窗口界面的方法之续篇

    在winform下实现左右布局多窗口界面的方法之续篇

    这篇文章主要介绍了在winform下实现左右布局多窗口界面的方法之续篇 的相关资料,需要的朋友可以参考下
    2016-02-02
  • 详解如何将.NET应用转换成Window服务

    详解如何将.NET应用转换成Window服务

    这篇文章主要为大家详细介绍了如何将.NET8.0应用程序转换成Windows服务,文中的示例代码讲解详细,有需要的小伙伴可以跟随小编一起学习一下
    2024-01-01
  • C# BeginInvoke实现异步编程方式

    C# BeginInvoke实现异步编程方式

    这篇文章主要介绍了C# BeginInvoke实现异步编程方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01

最新评论