当前位置: 首页 > news >正文

约瑟夫环递归算法详解与实现

一、引言

约瑟夫环问题是一个著名的理论问题,其背景是在古罗马时期,有n个犯人被围成一个圈,从第一个人开始报数,每次报到m的人将被处决,然后从下一个人开始重新报数,直到所有人都被处决。这个问题可以用递归算法来解决,本文将详细介绍约瑟夫环问题的递归算法,并给出C++代码实现。

二、约瑟夫环问题描述

有n个人围成一圈,从第一个人开始报数,每次报到m的人将被淘汰出局,然后从下一个人开始重新报数,直到最后只剩一个人为止。这个最后留下来的人被称为“幸运者”。我们需要找出这个幸运者的位置(即他在初始排列中的序号)。

三、递归算法思想

对于约瑟夫环问题,我们可以使用递归算法来解决。递归算法的基本思想是将问题分解为更小的子问题,然后逐个解决这些子问题,最终得到原问题的解

在约瑟夫环问题中,我们可以将原问题分解为n-1个人的子问题。假设我们知道在n-1个人中,谁是幸运者(即他在子问题中的序号),那么我们可以通过这个信息来找出在n个人中谁是幸运者。

具体来说,我们可以这样想:在n个人中,第一个被淘汰的人是第m个人(从第一个人开始报数)。当这个人被淘汰后,剩下的n-1个人形成了一个新的圈。这个新圈中的第一个人(即原圈中的第m+1个人)成为了新的起点。然后,我们在这个新圈中继续执行约瑟夫环问题的规则,直到找到幸运者。

在递归过程中,我们需要记录两个关键信息:一是当前圈中的人数n,二是报数的步长m。这两个信息将作为递归函数的参数传递给下一层递归。

四、递归算法实现

#include <iostream>
using namespace std;// 递归函数,返回在n个人中,报数步长为m时的幸运者的序号
int josephus(int n, int m) {if (n == 1) {// 当只剩下一个人时,他就是幸运者return 1;} else {// 否则,计算在第n个人被淘汰后,新的圈中幸运者的序号// 新的圈中的人数为n-1,报数步长仍为m// 由于第m个人被淘汰,所以新的圈中的幸运者在原圈中的序号为 (josephus(n-1, m) + m - 1) % n + 1// 注意:这里使用了取模运算和加1操作,以确保序号在1到n之间return (josephus(n-1, m) + m - 1) % n + 1;}
}int main() {int n, m;cout << "请输入总人数n和报数步长m:" << endl;cin >> n >> m;// 调用递归函数求解幸运者的序号int survivor = josephus(n, m);cout << "幸运者的序号是:" << survivor << endl;return 0;
}

五、算法分析

  1. 时间复杂度:递归算法的时间复杂度与递归的深度有关。在约瑟夫环问题中,递归的深度为n(总人数),因此时间复杂度为O(n)。虽然递归算法在某些情况下可能不如迭代算法高效,但它提供了更清晰的思维方式和更简洁的代码实现。
  2. 空间复杂度:递归算法的空间复杂度主要由递归栈的深度决定。在约瑟夫环问题中,递归栈的深度也为n(总人数),因此空间复杂度为O(n)。然而,在实际应用中,由于现代计算机的内存资源非常丰富,这个空间复杂度通常是可以接受的。

六、递归算法的优化

虽然上述递归算法已经能够解决约瑟夫环问题,但在某些情况下,我们可能希望进一步优化算法的性能。以下是一些可能的优化方法:

  1. 尾递归优化:在某些编程语言中(如C++),递归函数在调用自身时可能会产生额外的栈空间开销。为了减少这种开销,我们可以使用尾递归优化技术。尾递归优化允许编译器在调用递归函数时重用当前函数的栈帧,从而节省空间。然而,需要注意的是,C++标准并未强制要求编译器实现尾递归优化,因此在实际应用中可能无法获得预期的效果。
  2. 迭代算法:与递归算法相比,迭代算法通常具有更低的时间复杂度和空间复杂度。对于约瑟夫环问题,我们可以使用迭代算法来求解幸运者的序号。迭代算法的基本思想是使用一个循环来模拟报数和淘汰的过程,直到只剩下一个人为止。这种算法的时间复杂度和空间复杂度均为O(n),但在实际应用中通常具有更好的性能表现。

七、迭代算法实现

#include <iostream>
#include <vector>
using namespace std;// 迭代算法求解约瑟夫环问题
int josephusIterative(int n, int m) {if (n == 0 || m == 0) {return -1; // 输入无效,返回-1表示错误}vector<int> circle(n);for (int i = 0; i < n; ++i) {circle[i] = i + 1; // 初始化环,从1开始编号}int index = 0; // 当前位置索引while (n > 1) {// 向前移动m-1步for (int i = 0; i < m - 1; ++i) {index = (index + 1) % n;}// 淘汰当前位置的人cout << "淘汰的人序号是:" << circle[index] << endl;circle[index] = 0; // 标记为已淘汰// 向前移动一步到下一个位置index = (index + 1) % n;// 更新剩余人数n--;// 压缩环,将所有非零元素向前移动int j = 0;for (int i = 0; i < circle.size(); ++i) {if (circle[i] != 0) {circle[j++] = circle[i];}}// 更新环的大小circle.resize(j);}// 返回幸运者的序号for (int i = 0; i < circle.size(); ++i) {if (circle[i] != 0) {return circle[i];}}return -1; // 如果找不到幸运者,返回-1表示错误(实际上这种情况不会发生)
}int main() {int n, m;cout << "请输入总人数n和报数步长m:" << endl;cin >> n >> m;// 调用迭代算法求解幸运者的序号int survivor = josephusIterative(n, m);cout << "幸运者的序号是:" << survivor << endl;return 0;
}

八、算法对比

  1. 递归算法:递归算法具有简洁的代码实现和清晰的思维方式,但在处理大规模数据时可能会导致栈溢出或性能下降。此外,递归算法的空间复杂度通常较高,因为需要存储每一层递归的局部变量和返回地址。

  2. 迭代算法:迭代算法通过循环来模拟报数和淘汰的过程,避免了递归调用带来的额外开销。迭代算法的时间复杂度和空间复杂度均为O(n),且在实际应用中通常具有更好的性能表现。此外,迭代算法还可以方便地处理大规模数据,而无需担心栈溢出的问题。

九、结论

本文介绍了约瑟夫环问题的递归算法和迭代算法,并给出了相应的C++代码实现。递归算法具有简洁的代码实现和清晰的思维方式,但在处理大规模数据时可能存在性能问题。迭代算法通过循环来模拟报数和淘汰的过程,避免了递归调用的额外开销,并在实际应用中通常具有更好的性能表现。因此,在实际应用中,我们可以根据问题的规模和需求选择合适的算法来解决约瑟夫环问题。

相关文章:

约瑟夫环递归算法详解与实现

一、引言 约瑟夫环问题是一个著名的理论问题&#xff0c;其背景是在古罗马时期&#xff0c;有n个犯人被围成一个圈&#xff0c;从第一个人开始报数&#xff0c;每次报到m的人将被处决&#xff0c;然后从下一个人开始重新报数&#xff0c;直到所有人都被处决。这个问题可以用递…...

互联网应用主流框架整合之构建REST风格的系统

REST&#xff08;Representational State Transfer&#xff09;&#xff0c;中文译为“表述性状态转移”&#xff0c;是由Roy Fielding博士在他的博士论文中提出的一种软件架构风格&#xff0c;特别适用于网络应用的设计。REST不是一个标准&#xff0c;而是一种设计原则和约束集…...

vue3-自定义指令来实现input框输入限制

文章目录 前言具体实现分析主要部分详细解析导入和类型定义mounted 钩子函数unmounted 钩子函数指令注册使用 总结 前言 使用vue中的自定义指令来实现input框输入限制 其中关键代码强制触发input &#xff0c;来避免&#xff0c;输入规则外的字符时&#xff0c;没触发vue的响…...

MySQL日志——redolog

redo log&#xff08;重做日志&#xff09; 为什么需要redo log&#xff1f; 在mysql提交一个事务后&#xff0c;这个事务所作的数据修改并不会直接保存到磁盘文件中&#xff0c;而是先保存在buffer pool缓冲区中&#xff0c;在需要读取数据时&#xff0c;先从缓冲区中找&…...

Python热涨落流体力学求解算法和英伟达人工智能核评估模型

&#x1f3af;要点 &#x1f3af;平流扩散简单离散微分算子 | &#x1f3af;相场模拟&#xff1a;简单旋节线分解、枝晶凝固的 | &#x1f3af;求解二维波动方程&#xff0c;离散化时间导数 &#x1f3af;英伟达 A100 人工智能核性能评估模型 | &#x1f3af;热涨落流体动力学…...

【C语言】数组参数和指针参数详解

在写代码的时候难免要把【数组】或者【指针】传给函数&#xff0c;那函数的参数该如何设计呢&#xff1f; 1 一维数组传参 #include <stdio.h> void test(int arr[])//ok? {} void test(int arr[10])//ok? {} void test(int* arr)//ok? {} void test2(int* arr[20])…...

Tuple 元组

文章目录 一、什么是元组 &#xff1f;二、元组的具体操作2.1 创建元组2.1.1 tuple() 创建元组函数和 list() 创建列表函数总结 2.2 元组的元素访问操作2.3 元组的元素计数操作2.4 zip 对象 一、什么是元组 &#xff1f; 列表属于可变序列,可以任意修改列表中的元素。 元组的…...

(资料收藏)王阳明传《知行合一》共74讲,王阳明知行合一音频讲解资料

今天给大家带来的不是软件&#xff0c;而是一份精神食粮——《知行合一》的教程福利。这可不是一般的教程&#xff0c;它关乎心灵&#xff0c;关乎智慧&#xff0c;关乎我们如何在纷繁复杂的世界中找到自己的位置。 咱们得聊聊王阳明&#xff0c;这位明代的大儒&#xff0c;他…...

空气质量预报模式系统WRF-CMAQ

空气污染问题日益受到各级政府以及社会公众的高度重视&#xff0c;从实时的数据监测公布到空气质量数值预报及预报产品的发布&#xff0c;我国在空气质量监测和预报方面取得了一定进展。随着计算机技术的高速发展、空气污染监测手段的提高和人们对大气物理化学过程认识的深入&a…...

Collections.sort()方法总结

Collections.sort()方法总结 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;今天我们来探讨 Java 中的 Collections.sort() 方法。这个方法是 Java 集合框架中的…...

Java23种设计模式(二)

1、单例模式 单例模式&#xff08;Singleton Pattern&#xff09;是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式&#xff0c;它提供了一种创建对象的最佳方式。 这种模式涉及到一个单一的类&#xff0c;该类负责创建自己的对象&#xff0c;同时确保只有…...

Web前端收入来源:探索多元化的盈利渠道

Web前端收入来源&#xff1a;探索多元化的盈利渠道 在数字化时代&#xff0c;Web前端技术日益成为推动互联网业务发展的重要力量。对于前端开发者而言&#xff0c;除了传统的薪资收入外&#xff0c;还存在多种潜在的收入来源。本文将从四个方面、五个方面、六个方面和七个方面…...

抽象工厂模式(大话设计模式)C/C++版本

抽象工厂模式 C 参考&#xff1a;https://www.cnblogs.com/Galesaur-wcy/p/15927110.html #include <iostream> using namespace std;// 抽象产品Department ,定义具体产品的公共接口 class Department { public:virtual ~Department() default;virtual void Insert()…...

springboot宠物医院信息管理系统-计算机毕业设计源码04164

摘 要 现如今在中国&#xff0c;随着人民生活质量的逐渐提高&#xff0c;以及人民群众消费能力的日渐增长&#xff0c;各种各样的家养小动物&#xff0c;已经逐渐成为人类越来越亲密的生活伴侣。并且&#xff0c;现如今社会竞争及其激烈&#xff0c;人们的生活节奏越发急促、紧…...

Leetcode Hot100之哈希表

1. 两数之和 题目描述 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数&#xff0c;并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。但是&#xff0c;数组中同一个元素在答案里不能重复出现思路…...

Vision Transformer with Sparse Scan Prior

摘要 https://arxiv.org/pdf/2405.13335v1 In recent years, Transformers have achieved remarkable progress in computer vision tasks. However, their global modeling often comes with substantial computational overhead, in stark contrast to the human eye’s eff…...

笔记-python 中BeautifulSoup入门

在前面的例子用&#xff0c;我用了BeautifulSoup来从58同城抓取了手机维修的店铺信息&#xff0c;这个库使用起来的确是很方便的。本文是BeautifulSoup 的一个详细的介绍&#xff0c;算是入门把。文档地址&#xff1a;http://www.crummy.com/software/BeautifulSoup/bs4/doc/ …...

Tomcat Websocket应用实例研究

概述 本文介绍了如何根据Tomcat给出的websocket实例&#xff0c;通过对实例的学习&#xff0c;定制自己基于websocket的应用。 环境及版本&#xff1a; Ubuntu 22.04.4 LTSApache Tomcat/10.1.20openjdk 11.0.23 2024-04-16浏览器&#xff1a;Chrome 相关资源及链接 Class…...

leetcode-11-二叉树前中后序遍历以及层次遍历

一、递归版 前序遍历 &#xff08;先根遍历&#xff09; 中左右 class Solution {public List<Integer> preorderTraversal(TreeNode root) {List<Integer> result new ArrayList<Integer>();preorder(root, result);return result;}public void preorder…...

Python基础学习笔记(十一)——集合

目录 一、集合的介绍与创建二、集合的存储原理三、元素的修改1. 添加元素2. 删除元素 四、集合的运算五、集合的判定 一、集合的介绍与创建 集合&#xff08;set&#xff09;&#xff0c;一种可变、无序、不重复的数据结构&#xff0c;由大括号{}内、用逗号分隔的一组元素组成。…...

FineReport

1.FineReport 官网 &#xff1a;FineReport产品简介- FineReport帮助文档 - 全面的报表使用教程和学习资料 下载地址 免费下载FineReport - FineReport报表官网 FineReport是一款用于报表制作&#xff0c;分析和展示的工具。 普通模板&#xff1a;是 FineReport 最常用&#xf…...

嵌入式就业前景好么

嵌入式就业前景在当前环境下是较为乐观的&#xff0c;以下是对嵌入式就业前景的详细分析&#xff1a; 广泛应用领域&#xff1a;嵌入式系统广泛应用于智能家居、医疗设备、航空航天等领域。随着物联网&#xff08;IoT&#xff09;的快速发展&#xff0c;预计到2024年&#xff…...

为啥找对象千万别找大厂男,还好我不是大厂的。。

网上看到一大厂女员工发文说&#xff1a;找对象千万别找大厂男&#xff0c;理由说了一大堆&#xff0c;无非就是大厂男为了逃避带娃&#xff0c;以加班为由宁愿在工位上玩游戏也不愿回家。当然这种观点有的人赞同有的人反对。 网友精彩评论&#xff1a; --------------下面是今…...

如何查看k8s中service的负载均衡策略

在Kubernetes中&#xff0c;Service的负载均衡策略一般由kube-proxy负责&#xff0c;kube-proxy使用iptables或IPVS规则进行负载均衡。默认情况下&#xff0c;kube-proxy使用的是轮询&#xff08;Round Robin&#xff09;策略&#xff0c;但是在使用IPVS模式时&#xff0c;可以…...

Linux-DNS域名解析服务01

BIND 域名服务基础 1、DNS&#xff08;Domain Name System&#xff09;系统的作用及类型 整个 Internet 大家庭中连接了数以亿计的服务器、个人主机&#xff0c;其中大部分的网站、邮件等服务器都使用了域名形式的地址&#xff0c;如 www.google.com、mail.163.com 等。很显然…...

[c++刷题]贪心算法.N01

题目如上: 首先通过经验分析&#xff0c;要用最少的减半次数&#xff0c;使得数组总和减少至一半以上&#xff0c;那么第一反应就是每次都挑数组中最大的数据去减半&#xff0c;这样可以是每次数组总和值减少程度最大化。 代码思路:利用大根堆去找数据中的最大值&#xff0c;…...

推荐常用的三款源代码防泄密软件

三款源代码防泄密软件——安秉源代码加密、Virbox Protector 和 MapoLicensor——确实各自在源代码保护的不同方面有其专长。这些软件可以满足企业对于源代码保护的三大需求&#xff1a;防止泄露、防止反编译和防止破解。 安秉源代码加密&#xff1a; 专注于源代码文件的加密&…...

Android 13 高通设备热点低功耗模式(2)

前言 之前写过一篇文章:高通热点被IOS设备识别为低数据模式,该功能仿照小米的低数据模式写的,散发的热点可以达到被IOS和小米设备识别为低数据模式。但是发现IOS设备如果后台无任何网络请求的时候,息屏的状态下过一会,会自动断开热点的连接。 分析 抓取设备的热点相关的…...

web前端任职条件:全面解析

web前端任职条件&#xff1a;全面解析 在当今数字化快速发展的时代&#xff0c;Web前端技术已经成为互联网行业不可或缺的一部分。作为一名Web前端开发者&#xff0c;需要具备哪些任职条件呢&#xff1f;本文将从四个方面、五个方面、六个方面和七个方面为您深入剖析。 四个方…...

分析医药零售数据该用哪个BI数据可视化工具?

数据是企业决策的重要依据&#xff0c;可以用于现代企业大数据可视化分析的BI工具有很多&#xff0c;各有各擅长的领域。那么哪个BI数据可视化工具分析医药零售数据又好又快&#xff1f; 做医药零售数据分析首推奥威BI数据可视化工具&#xff01; 奥威BI数据可视化工具做医药…...

工商局网站怎么做增项/足球世界排名国家

【摘要】PHP即“超文本预处理器”&#xff0c;是一种通用开源脚本语言。PHP是在服务器端执行的脚本语言&#xff0c;与C语言类似&#xff0c;是常用的网站编程语言。PHP独特的语法混合了C、Java、Perl以及 PHP 自创的语法。下面是php怎么去掉字符串末尾字符&#xff0c;让我们一…...

公司网站建设是什么费用/厦门seo哪家强

Servicemix的优点&#xff1a; 1&#xff0c;基于JBI规范&#xff1b; 2&#xff0c;可以热部署&#xff1b; 3&#xff0c;支持Camel&#xff08;可以用DSL去开发集成流程&#xff09;&#xff1b; Servicemix的缺点&#xff1a; 1&#xff0c;JBI规范带来了使用上的繁琐…...

张家港网站建设/创建一个网站

首先&#xff0c;先考虑一个问题&#xff0c;什么条件会触发cancelAcquire()方法&#xff1f; cancelAcquire()方法的反向查找可以清楚的看到在互斥锁和共享锁的拿锁过程中都是有调用此方法的&#xff0c;而cancelAcquire()方法是写在finally代码块中&#xff0c;并且使用faile…...

企业网站能个人备案吗/seo排名计费系统

北京国际版权交易中心聚集整合各类版权专业服务资源、数字阅读类领军企业盛大文学及旗下各网站、中文在线、搜狐读书、新浪图书、腾讯图书等13家主流阅读网站联合发出倡议&#xff1a;将每年的10月26日设立为“数字阅读日”&#xff0c;倡导在线“健康阅读”、 “主题阅读”及 …...

把做的网站发布打万维网上/最好用的手机优化软件

本节书摘来自异步社区《Adobe SiteCatalyst网站分析权威手册》一书中的第1章&#xff0c;第1.2节&#xff0c;作者【美】Adam Greco&#xff0c;更多章节内容可以访问云栖社区“异步社区”公众号查看 1.2 SiteCatalyst在组织中所处的角色 Adobe SiteCatalyst网站分析权威手册我…...

linux下载wordpress/保定网站seo

姓名年龄性别职位死亡时间所在行业死亡原因注信息来源王江民59岁男酷6网研发部 软件工程师2010.4.4IT心脏病死亡时仅 入职3个月百度百科罗耀明80后男江民创始人 兼总裁2009.11.0IT急性病毒 性心肌炎加班 彭小琦30岁左右男搜狐无线事业部 技术人员2010.4.0IT过度劳累 导致猝死工…...