一、简单的约瑟环夫

1.故事描述:

据说著名犹太历史学家 Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓到,于是决定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。然而Josephus 和他的朋友并不想遵从,但是最后却躲过了自杀。请问,他们是怎么做到的?

2.算法分析:

一个明显的求解方法便是将41个人排成一个环,内圈为按照逆时针的最初编号,外圈为每个数到数字3的顺序,如图。我们可以使用数组来保存约瑟夫环中该自杀的数据,而数组的下标作为参与人员的编号,并将数组看做为环形来处理.

根据上面的思路,编写代码:

//简单约瑟夫环求解
#include <iostream>
#include <time.h>
#include <string.h>#define Num 41
#define KillMan 3void Josephus(int alive)
{int man[Num] = {0};int count = 1;int i = 0,pos = -1;while (count <= Num){do {pos = (pos + 1) % Num;if (man[pos] == 0)i++;if (i == KillMan){i = 0;break;}} while (1);man[pos] = count;printf("第%2d个人自杀!约瑟夫环编号为%2d",pos+1,man[pos]);if (count % 2){printf("->");}else{printf("->\n");}count++;}printf("\n这%需要存活的人初始位置应排在以下序号:\n",alive);alive = Num -alive;for (i = 0; i<Num; i++){if (man[i] > alive){printf("初始编号:%d,约瑟夫环编号:%d\n",i+1,man[i]);}}printf("\n");
}int main(int argc, const char * argv[])
{// std::cout << "Hello, World!\n";int alive;printf("约瑟夫环问题求解!\n");printf("输入需要留存的人的数量:");scanf("%d",&alive);Josephus(alive);return 0;
}

演算结果:

在这里,Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏。- -

二、复杂约瑟夫环算法

问题描述:

在控制台下实现约瑟夫环:编号为1,2,3,……,n 的 n 个人按顺时针方向围坐一圈,每人持有一个密码(正整数)。一开始任选一个正整数作为报数上限值 m,从第一个人开始按顺时针方向自1开始报数,报到 m时停止报数。报m的人出列,将他的密码作为新的 m 值,从他在顺时针方向上的下一个人开始重新报数,如此下去,直至所有人全部出列为止。

我们先分析一这个下问题,同前面的约瑟夫环相比明显复杂得多,主要体现在如下几点:

  • 这里,参考人的个数为n是一个可变量,因此,不能使用数组来表示,应该使用链表结构来表示.另外,这n个人首尾相接构成一个环,因此应该使用循环链表处理这个问题。
  • 约瑟夫环的出列数字不是固定值,而是每个人具有不同的值。这要在程序中分别处理。
其它部分和前述的约瑟夫环类似。我们可以参照这里的分析和前述的算法来编写这个复杂约瑟夫环的求解算法.代码如下:
#include <iostream>
#include <time.h>
#include <string.h>typedef struct node
{int number;         //游戏者编号int psw;            //出列数字struct node *next;
}LNode,*LinkList;void insertList(LinkList *list,LinkList q,int number,int psw)   //添加新结点
{LinkList p;p = (LinkList)malloc(sizeof(LNode));p->number = number;p->psw = psw;if (!*list){*list = p;p->next = NULL;}else{p->next = q->next;q->next = p;}}void CircleFun(LinkList *list,int m)    //算法
{LinkList p,q;int i;q = p = *list;while (q->next != p){q = q->next;}printf("游戏者按照如下顺序出列.\n");while (p->next != p){for (i = 0; i<m-1; i++){q = p;p = p->next;}q->next = p->next;  //删除p指向的结点printf("第%d个人出列,其手中的出列数字为%d.\n",p->number,p->psw);m = p->psw;         //重置出数列字free(p);p = q->next;}printf("\n最后一个人是%d,其手中的出列数字为%d.",p->number,p->psw);
}int main(int argc, const char * argv[])
{// std::cout << "Hello, World!\n";int alive;LinkList list1 = NULL,q= NULL,list;int num,baoshu;int i,e;printf("约瑟夫环问题求解!\n");printf("输入约瑟夫环中的人数:\n");scanf("%d",&num);printf("按照顺序输入每个人手中的出列数字:\n");for (i= 0; i<num; i++){scanf("%d",&e);insertList(&list1, q, i+1, e);if (i == 0){q = list1;}else{q = q->next;}}q->next = list1;list = list1;printf("请输入第一次出列的数字:\n");scanf("%d",&baoshu);CircleFun(&list, baoshu);printf("\n");return 0;
}

演算结果:

参考书籍:《C/C++常用算法手册》

c/c++常用算法(14) -- 经典数据结构(约瑟环夫问题)相关推荐

  1. AI之NLP:自然语言处理技术简介(是什么/学什么/怎么用)、常用算法、经典案例之详细攻略(建议收藏)

    AI之NLP:自然语言处理技术简介(是什么/学什么/怎么用).常用算法.经典案例之详细攻略(建议收藏) 目录 NLP是什么? 1.NLP前置技术解析 2.python中NLP技术相关库 3.NLP案例 ...

  2. AI之NLP:自然语言处理技术简介(是什么/学什么/怎么用)、常用算法、经典案例之详细攻略(建议收藏)daiding

    AI之NLP:自然语言处理技术简介(是什么/学什么/怎么用).常用算法.经典案例之详细攻略(建议收藏) 目录 NLP是什么? 1.NLP前置技术解析 2.python中NLP技术相关库 3.NLP案例 ...

  3. 约瑟环的python实现(举例说明)

    题目是: N个人围成一圈报数,报到某一个数m的就出局,问你最后剩下来的人的号码? 网上通用约瑟夫环的算法是: //函数接收n和m,返回最后出圈的是第几个人 /*e.g. yuesefu(5,2)=3 ...

  4. 约瑟环问题JAVA(数组)

    所有人围成一圈,顺序排号.从第一个人开始报数(从 1 到 3 报数), 凡报到 3 的人退出圈子,剩下的人继续报数,最后留下的当选为队长. 请你通过编写程序,求出一组人中的队长是原来第几位同学.(约瑟 ...

  5. Josephus约瑟环问题

    约瑟环问题描述:n个人有序围成一圈报数,报到M的就退(自)出(杀),再从头开始报数,由此循环,求剩下最后一人的序号. 第一种直觉的想法是弄个list循环: int Josephus(unsigned ...

  6. 约瑟环问题 Java解法

    约瑟环问题: 17世纪的法国数学家加斯帕在<数目的游戏问题>中讲的一个故事,15个教徒和15个非教徒在深海上遇险,必须将一半的人投入海中,其余的人才能幸免于难,于是想了一个办法,30个人围 ...

  7. 约瑟环问题(全部代码在本文下方)

    用循环链表编程实现约瑟夫问题,n个人围成一圈,从某人开始报数1,2,.., m,数到m的人出圈,然后从出圈的下一个人(m+1)开始重复此过程,直到全部人出圈,于是得到一个出圈人员的新序列,如当n=8, ...

  8. Offer 62. 约瑟环(规律问题-递归-中等)

    Offer 62. 圆圈中最后剩下的数字 思路1:⭐ 创建一个下标为 0~n-1的数组 | 当n=5时 nums = [1 1 1 1 1] 代表下标为i的数是否已经被删除⭐ 通过不断(外层设置一个循 ...

  9. python常用代码_Python常用算法学习(4) 数据结构(原理+代码)-最全总结

    数据结构简介 1,数据结构 数据结构是指相互之间存在着一种或多种关系的数据元素的集合和该集合中数据元素之间的关系组成.简单来说,数据结构就是设计数据以何种方式组织并存贮在计算机中.比如:列表,集合与字 ...

最新文章

  1. 为什么使用HashMap需要重写hashcode和equals方法_为什么要重写 hashcode 和 equals 方法?...
  2. 【SpringBoot】如何在静态工具类中注入mapper?
  3. [Java基础]接口组成(默认方法,静态方法,私有方法)
  4. LINQ to JavaScript
  5. 2012年下半年我国信息化和信息安全走势分析与预测
  6. Machine Learning ——Homework3
  7. 将时间戳转化为带时区的格式
  8. java实现modbus rtu协议与 modscan等工具(3)物理连接
  9. wps表格宏被禁用如何解禁_怎么解除宏禁用_win7系统取消宏已被禁用提示的方法...
  10. SharePoint 2016 Beta 2 使用体验
  11. 高仿网易评论列表效果之界面生成
  12. OpenDDS的动态绑定bind_config传输配置和实例
  13. ValueError: Object arrays cannot be loaded when allow_pickle=False 异常的多种处理方式
  14. CVE-2020-25540:ThinkAdmin未授权列目录/任意文件读取漏洞复现
  15. javaweb大全:地址来自 孤傲苍狼
  16. UV、PV、IP意思及区别解释
  17. 展览会议签到方式之——CES亚洲消费电子展自助签到
  18. Visual Studio 2019 代码显示空格等空白符
  19. Windowsnbsp;XP/Vista/Windows7常见蓝…
  20. 海康摄像头http抓图

热门文章

  1. 2017湘潭大学邀请赛H题(树的直径)
  2. Symbian模拟器“Cannot start ECMT Manager”错误提示解决方案
  3. 老年食堂:让社区生活充满“幸福滋味”
  4. 【Bellhop-二维海水声速环境下的声线仿真】
  5. 十六届智能车全向组算法开源(一)
  6. 2023新版PMP考试有哪些变化?
  7. Visual C++ Tips: warning C4996: 'stricmp' was declared deprecated
  8. 《李笑来通往财富自由之路》的学习笔记(完)
  9. conda安装SimpleITK和pydicom
  10. 软件下载官网系统源码