约瑟夫环公式(约瑟夫环公式数学推导)

http://www.itjxue.com  2023-01-25 19:23  来源:未知  点击次数: 

约瑟夫环问题

问题描述:n个人(编号0~(n-1)),从0开始报数,报到(m-1)的退出,剩下的人继续从0开始报数。求胜利者的编号。

我们知道第一个人(编号一定是m mod n-1) 出列之后,剩下的n-1个人组成了一个新的约瑟夫环(以编号为k=m mod n的人开始):

k k+1 k+2 ... n-2, n-1, 0, 1, 2, ... k-2

并且从k开始报0。

现在我们把他们的编号做一下转换:

k -- 0

k+1 -- 1

k+2 -- 2

...

...

k-2 -- n-2

k-1 -- n-1

变换后就完完全全成为了(n-1)个人报数的子问题,假如我们知道这个子问题的解:例如x是最终的胜利者,那么根据上面这个表把这个x变回去不刚好就是n个人情况的解吗?!!变回去的公式很简单,相信大家都可以推出来:x'=(x+k) mod n

如何知道(n-1)个人报数的问题的解?对,只要知道(n-2)个人的解就行了。(n-2)个人的解呢?当然是先求(n-3)的情况 ---- 这显然就是一个倒推问题!好了,思路出来了,下面写递推公式:

令f表示i个人玩游戏报m退出最后胜利者的编号,最后的结果自然是f[n]

递推公式

f[1]=0;

f=(f+m) mod i; (i1)

有了这个公式,我们要做的就是从1-n顺序算出f的数值,最后结果是f[n]。因为实际生活中编号总是从1开始,我们输出f[n]+1

C语言约瑟夫环问题中这个 i = (i + m - 1) %n公式是怎么把3 6 9之类的删掉的

i = (i + m - 1) %n 只是用于循环内部,i的赋值(即用于计算下一个i值);

请参考下方链接中的解法一

网页链接

约瑟夫环问题 经典求循环公式

提问时 应该把问题说清,不要只帖代码

要中说查错的话:

cout "The winner is No."index".\n";//少了个后引号

while(countM)// 对成功的报数开始计数

{

index=(index+1)%N;//计算要报数的小孩编号 这步谁能帮我分析下?

if (in_circle[index])count++;

}

这个循环的作用就是让还在环中的小孩报数,考虑到报到队尾时必须回到头再重报,所以就有了

index=(index+1)%N;

举例来说:

从队尾index=19回队头index=0

第一圈报数时,第20个小孩(index==19)报count==5然后出圈了//if (in_circle[index])count++;

再持行index=(index+1)%N;时即从队头开始.

第二圈报数时,到第19(index==18)个小孩报数count==1后,

再持行index=(index+1)%N//index==19

if...//不计数

index=(index+1)%N//回到第一个小孩

if....//第一个小孩报count==2

约瑟夫环问题的第推公式是怎么来的阿~~

/*约瑟夫问题的数学方法

无论是用链表实现还是用数组实现都有一个共同点:要模拟整个游戏过程,不仅程序写起来比较烦,而且时间复杂度高达O(nm),当n,m非常大(例如上百万,上千万)的时候,几乎是没有办法在短时间内出结果的。我们注意到原问题仅仅是要求出最后的胜利者的序号,而不是要读者模拟整个过程。因此如果要追求效率,就要打破常规,实施一点数学策略。

为了讨论方便,先把问题稍微改变一下,并不影响原意:

问题描述:n个人(编号0~(n-1)),从0开始报数,报到(m-1)的退出,剩下的人继续从0开始报数。求胜利者的编号。

我们知道第一个人(编号一定是m%n-1) 出列之后,剩下的n-1个人组成了一个新的约瑟夫环(以编号为k=m%n的人开始):

k k+1 k+2 ... n-2, n-1, 0, 1, 2, ... k-2

并且从k开始报0。

现在我们把他们的编号做一下转换:

k -- 0

k+1 -- 1

k+2 -- 2

...

...

k-2 -- n-2

k-1 -- n-1

变换后就完完全全成为了(n-1)个人报数的子问题,假如我们知道这个子问题的解:例如x是最终的胜利者,那么根据上面这个表把这个x变回去不刚好就是n个人情况的解吗?!!变回去的公式很简单,相信大家都可以推出来:x'=(x+k)%n

如何知道(n-1)个人报数的问题的解?对,只要知道(n-2)个人的解就行了。(n-2)个人的解呢?当然是先求(n-3)的情况 ---- 这显然就是一个倒推问题!好了,思路出来了,下面写递推公式:

令f[i]表示i个人玩游戏报m退出最后胜利者的编号,最后的结果自然是f[n]

递推公式

f[1]=0;

f[i]=(f[i-1]+m)%i; (i1)

有了这个公式,我们要做的就是从1-n顺序算出f[i]的数值,最后结果是f[n]。因为实际生活中编号总是从1开始,我们输出f[n]+1

由于是逐级递推,不需要保存每个f[i],程序也是异常简单:*/

#include stdio.h

main()

{

int n, m, i, s=0;

printf("N="); scanf("%d", n);

printf("M="); scanf("%d", m);

for(i=2; i=n; i++) s=(s+m)%i;

printf("The winner is %d\n", s+1);

}

/*这个算法的时间复杂度为O(n),相对于模拟算法已经有了很大的提高。算n,m等于一百万,一千万的情况不是问题了。可见,适当地运用数学策略,不仅可以让编程变得简单,而且往往会成倍地提高算法执行效率。*/

有n个人围成一圈,顺序排号。凡报到3的人退出圈子,问最后留下的是原来第几号的那位?

这是约瑟夫环的数学解法(迭代法)的公式,我们可以这样理解

把n个人想成标号从0开始到n-1的n个人,报到3的人退出圈子,

那么退出圈子的人在0到n-1的标号为(k+3)%n(其中k为n-1个人时退出圈子的人的标号)

因为有一个人退出了圈子,所以还剩下n-1个人,我们对剩下的人重新从0到n-2编号,

同样有公式(k1+3)%n(其中k1为n-2个人时退出圈子的人的标号)得出n-1个人时退出圈子人的标号,

以此类推直到n等于1时kn-1=0也就是1个人时留下的就是标号为0的人

以此有递推公式f(1)=0,f(i)=(f(i-1)+3)%i f(i)为第i次退出圈子的人

我们用for循环从2个人时开始反推,经过n-1次迭代,去除退出圈子的人,

从而可以得到n个人时最后一个退出的人的标号为几,也就是最后留下的人的标号是几,

因为我们每次标号都是从0开始所以最后退出的人的实际编号=标号+1.

有了上述分析我们就不难得出程序(见图)

约瑟夫环公式是怎样推导出来的?

1、约瑟夫环公式推导:已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围。从编号为k的人开始报数,数到m的那个人出列;他的下一个人又从1开始报数,数到m的那个人又出列。

依此规律重复下去,直到圆桌周围的人全部出列。

这个就是约瑟夫环问题的实际场景,有一种是要通过输入n,m,k三个正整数,来求出列的序列。这个问题采用的是典型的循环链表的数据结构,就是将一个链表的尾元素指针指向队首元素。 p-link=head。

2、解决问题的核心步骤:

1.建立一个具有n个链结点,无头结点的循环链表。

2.确定第1个报数人的位置。

3.不断地从链表中删除链结点,直到链表为空。

扩展资料

算法例子

C#

//1、循环链表存储结构? ?

class LinkData

{

public int value { get; set; }//小孩子的ID

public LinkData next { get; set; }//下一个小孩子的位置?

private LinkData(int m_value)

{

value=m_value;

}? ? ?

//孩子们围坐一圈

public static LinkData CreateLink(int []arr)

{

LinkData head = new LinkData(0);

LinkData p = head;

for(int i=0;iarr.Length-1;i++)

{

p.value = arr[i];

p.next = new LinkData(0);

p = p.next;

}

p.value = arr[arr.Length - 1];

p.next = head;//循环链表,尾巴指向头

return head;

}

//丢手绢算法

public static void Yuesefu(LinkData head, int i, int M)

{

//DateTime dt = DateTime.Now;

//Console.WriteLine("link go:");

LinkData f = head;//头

LinkData r=f;//尾

for (; i 0; i--) //进入移动到第一次丢手绢的位置

{

r = f;

f = f.next;

}

while (r.next != r)//是否剩下最后一个小孩子

{

for(int j=0;jM;j++)

{

r=f;

f=f.next;

}

Console.Write(f.value.ToString() + " ");//小孩子报上名来

f = f.next;//踢掉一个小孩子

r.next = f;

}

Console.WriteLine(r.value.ToString());//小孩子报上名来

//Console.WriteLine(string.Format("耗时{0}毫秒",(DateTime.Now-dt).TotalMilliseconds));

}

}

//2、ListInt存储结构

class ListData

{

//丢手绢算法,直接通过在ListInt集合中定位元素,再移除元素,循环往复,直到集合为空

public static void Yuesefu(Listint src, int i, int M)

{

int len = src.Count;

i = (i + M) % src.Count;

//Console.WriteLine("list go:");

//DateTime dt = DateTime.Now;

while (src.Count 1)

{

Console.Write(src[i].ToString() + " ");//小孩子报上名来

src.RemoveAt(i);//踢掉一个小孩子

i = (i + M) % src.Count;

}

Console.WriteLine(src[i].ToString());//小孩子报上名来

//Console.WriteLine(string.Format("耗时{0}毫秒", (DateTime.Now - dt).TotalMilliseconds));

}

}

参考资料:百度百科——约瑟夫环

(责任编辑:IT教学网)

更多

推荐3DMAX教程文章