关于约瑟夫环问题的思考(数组做法)

2023-05-16

这几天做题时碰见了一个很有意思的问题,也是一个十分经典问题—约瑟夫环问题。问题很简单,就是有n个人围成一个圈,每隔m个人就自杀一个,直到剩下最后一个人为止,问最后剩下的最后一个人是谁?
来看具体的题目描述:这个是C语言网的ACM训练题----出圈
在这里插入图片描述
思路:这道题就是让你求解约瑟夫环问题,我在看到这道题时,第一反应是用数组做,用数组来构造一个环。实际上用数组来做没有任何问题,但我却碰到了一个使用for的一个坑,直接影响到了我整道题目,接下来我把代码分享出来,希望大家不要再踩这个坑
错误代码:

#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e4;
int a[maxn];
int main(){
int n,m;
while(scanf("%d %d",&n,&m)!=EOF){
    //这部分代码我是初始化一个1,2,3,4,5....的数组,他们的下标从0开始分别为0,1,2,4,...
    a[0] = 1;
    for(int i = 0;i<=n-2;i++){
        a[i+1] = a[i]+1;
    }
    //cnt使用来计数,当cnt==m时来选择剔除的的人
    int cnt =0;
    int sum = 0;//剔除的人数,当剔除了n-1个人时,就停止操作,循环的结束条件
    //开始清人,因为我在循环体内重新定义了i,所以这里我把i定义在外面,防止重复定义
    int i = 0;
    for(;i<n;i++){
        //剔除的数组元素,我把他赋值为-1
        if(a[i]!=-1){
            cnt++;
        }
        //隔了m个人,且没有被剔除出去,就把他赋值为-1把他剔除出去,cnt归零重新计数,sum加1
        if(cnt==m&&a[i]!=-1){
            a[i] = -1;
            cnt = 0;
            sum++;
        }

        if(i==n-1){
            //这是一个环,我们遍历完数组,还要重头开始(注意)
            i = 0;
        }
       //循环结束条件,当剔除了n-1个人,还剩一个人时,结束循环
       if(sum==n-1){
            break;
        }
}
//输出打印不是-1的,就是剩下的那个人的值
for(int i = 0;i<n;i++){
    if(a[i]!=-1){
        printf("%d\n",a[i]);
        break;
    }
}
}
return 0;
}

看到这儿,你可以发现其实思路是没问题的,但输出的结果是这样的
在这里插入图片描述
我们不管输入什么数据,它给我们的结果都是1,始终都是第一个人最后活下来,那究竟是为什么嘞?
首先我们要明白在循环体内重新对循环变量进行赋值,是有个特别要注意的事项:

int i = 0;
for(;i<n;i++){
   if(i==n-1){
        i = 0;
   }
}

这段代码就是我们用数组构造一个环的关键代码,当程序遍历完数组的最后一个元素时,我们把i重新赋值为0, 使它能像一个环一样能够被重复访问,这个基本的思路是没有问题的。但这里有个细节就是我重新赋值为0,进入下一层for循环时,满足我们for循环的条件,是要先执行i++的。换句话说,我们想要下一次的遍历从下标0开始,但实际上它是从下标1开始的,第一个数字在每次循环遍历都是被排除在外的,那么在约瑟夫环中最后剩下的肯定都是第一个元素。弄清楚错误的原因,我们不妨在初始化数组时就留一手(留一个缓冲位),如图:
以在5个人中隔3个人就剔除出去为例(n = 5,m =3)
在这里插入图片描述
我们在初始化数组时,就留出一个空位,也就说我们从下标1开始赋值,下标0的位置来做一个缓冲的作用。 我们遍历完最后一个元素后,将循环变量重置为下标0, 同样程序还是会先进行i++的操作,那么i 就等于1了,而我们的数字不就是从下标1赋值的吗,刚好合适,就没有把任何元素都排除在外了。
更改后的环代码:

//从下标1开始赋值
    for(int i = 1;i<=n;i++){
        a[i] = i;
    }
//初始的时候还是要从下标1开始的
int i = 1;
for(;i<=n;i++){
   if(i==n){
        i = 0;
   }
}

这样一来,我们对于这个问题数组做法最坑的一部分就解决了(至少我是这么认为),接下来无非就是用代码模拟这个题目要求的步骤,思路很简单,给出完整的AC代码:

#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e4;
int a[maxn];
int main(){
int n,m;
while(scanf("%d %d",&n,&m)!=EOF){
    for(int i = 1;i<=n;i++){
        a[i] = i;
    }
    int cnt =0;
    int sum = 0;//剔除的人数
    //开始清人
    int i = 1;
    for(;i<=n;i++){
        //被剔除出去的人已经不算在环里
        if(a[i]!=-1){
            cnt++;
        }
        //隔了m个人,开始剔除一个人
        if(cnt==m&&a[i]!=-1){
            a[i] = -1;
            cnt = 0;
            sum++;
        }

        if(i==n){
            //这是一个环
            i = 0;

        }
      //当剔除到只剩下一个人时,就退出循环
       if(sum==n-1){
            break;
        }
}
for(int i = 1;i<=n;i++){
    if(a[i]!=-1){
        printf("%d\n",a[i]);
        break;
    }
}
}
return 0;
}

根据这个题目,我们可以看出其实在很多题目中,隐藏了很多的细节,很多题AC不了,估计就是这些细节导致的,所以我们一定要注重细节上的东西。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

关于约瑟夫环问题的思考(数组做法) 的相关文章

  • 网页数据解析与爬取----Beautiful Soup

    目录 网页数据解析与爬取 Beautiful SoupBeautiful Soup 使用1 Beautiful Soup简介2 解析器3 准备工作4 节点选择器5 提取信息1 获取名称2 获取属性3 获取内容4 嵌套选择 6 关联选择1 子
  • FTP :身份验证、本地用户访问、虚拟用户访问实验

    ftp ftp文件共享 主要用于存储 xff0c 采用c s架构 xff0c 客户端可以通过登录server xff0c 去实现文件的上传 xff0c 删除的操作 ftp工作模式 主动传输模式 client使用N端口向ftp的server的
  • SQL server快捷键

    F5 选中文本调试或运行 Ctrl 43 K xff0c Ctrl 43 C xff1a 注释选定内容 Ctrl 43 K xff0c Ctrl 43 U xff1a 取消注释选定内容 Ctrl 43 K xff0c Ctrl 43 xff
  • 143-牛客网C++刷题9

    1 不定义第三个变量 xff0c 交换两个变量的数据 x 43 61 y y 61 x y x 61 y 2 常量可以是任何的基本数据类型 xff0c 可分为整型数字 浮点数字 字符 字符串和布尔值 3 在C 43 43 中 xff0c 可
  • GitHub|搭建个人静态页并绑定私有域名

    Ps xff1a 仅自学自用留档 xff0c 如有需要请自行找寻内容 xff01 使用Git在GitHub上搭建个人静态页并绑定域名 一 在GitHub创建特殊的个人仓库二 将仓库保存到本地并使用Git链接1 使用Git将仓库保存到本地文件
  • java Collection常用方法

    主要方法如下所述 然后我们来具体写一下这些方法 首先是add方法 span class token keyword public span span class token keyword static span span class to
  • java Collections基本概念和常用方法

    Collections是一个类 他在java的util包下 所以使用它是需要导包的 Collections是一个静态方法的集合类 他里面的方法都是静态的 Collections中的方法有很多 这里我们主要看三个 Collections的方法
  • 使用域控批量安装软件

    域自带的批量部署软件有多种方式 xff1a 1 xff0c 发布 xff0c 域服务器发布软件 xff0c 客户端到添加删除程序 添加新程序中点击安装 2 xff0c 分配指派到用户 xff0c 在客户端用户登录时自动安装 3 xff0c
  • XXL-JOB:com.fasterxml.jackson.databind.JsonMappingException: Unexpected character (‘o‘ (code 111))解决

    背景 项目中的xxl job admin版本为2 1 1 xff0c 一直运行的很好 xff0c 但是有一天被扫出安全漏洞 xff0c 然后 xff0c 我就把xxl job admin的springboot版本由1 5升级为2 2 1版本
  • Ubuntu部分图标缺失,包括部分系统图标

    ubuntu部分图标缺失 这里说的缺失不是指图标不会显示 xff0c 而是说图标虽然会显示 xff0c 但是显示不正确 比如显示为一个空白方块或者红色的 34 禁止 34 图标 简要列出部分缺失的图标 xff1a 文件夹图标wifi图标 x
  • python 添加图例_Python | 在图上添加图例

    python 添加图例 Adding legend is the best way to label data series plotted on a graph Matplotlib has an inbuilt defined func
  • java有哪些集合类型?集合类的特点

    Java属于入门容易 xff0c 天花板却极高的编程语言 java有哪些集合类型 对于java工程师来说技术的不断发展 需要不断学习java进阶知识 为了帮助大家巩固基础 xff0c 本文解答了java有哪些集合类型 集合类的特点是什么 x
  • MATLAB(一)基本操作与矩阵输入

    文章目录 前言一 Matlab视窗二 基本操作与矩阵输入1 把MATLAB当做计算机2 初等数学函数Exercise练习 2 嵌入函数3 特殊变量和常量4 MATLAB调用优先5 数字显示格式长Exercise练习 6 命令行终端7 部分函
  • MATLAB(六)图形界面_GUI_程式设计

    文章目录 前言MATLAB GUI Programs启动GUI程序对齐组件给按钮标上标签GUI脚本结构function untitled OpeningFcn对象的回调Set the axes for PlottingExercise练习P
  • Excel 精选28个技巧

    文章目录 前言1 一键求和2 一键插入柱形图3 单元格内强制换行4 快速移动资料5 快速生成下拉式功能表6 计算带单位的数据7 小写金额转大写8 快速输入 9 批量添加下划线10 文字随单元格大小变化11 图片随单元格大小变化12 快速提取
  • 调度算法——时间片轮转、优先级、多级反馈队列(例题详细!!!)

    文章目录 前言知识总览时间片轮转 xff08 RR Round Robin xff09 优先级调度算法多级反馈队列调度算法知识回顾与重要考点 前言 此篇文章是我在B站学习时所做的笔记 xff0c 大部分图片都是课件老师的PPT xff0c
  • 生产者-消费者问题(有例题!!!)

    文章目录 前言问题描述如何实现思考 xff1a 能否改变相邻P V操作的顺序 知识回顾与重要考点 前言 此篇文章是我在B站学习时所做的笔记 xff0c 大部分图片都是课件老师的PPT xff0c 方便复习用 此篇文章仅供学习参考 提示 xf
  • 计算机网络习题——循环冗余校验

    3 07 要发送的数据为1101011011 采用CRC的生成多项式是 P X 61 X 4 43 X 43 1 试求应添加在数据后面的余数 xff08 1 xff09 若要发送的数据在传输过程中最后一个1变成了0 xff0c 即变成了11
  • 计算机网络课后答案(谢希仁第八版)

    计算机网络课后答案 谢希仁第八版
  • linux系统 删除文件命令

    Linux系统下删除文件是一个非常高频的需求 xff0c 几乎每天都会遇到 xff0c 所以rm命令是一个非常常用Linux命令 rm命令是英文单词 remove 的缩写 xff0c 它主要作用是 xff1a 1 删除文件 xff1b 2

随机推荐

  • 常见的HTTP状态码列表

    HTTP状态码列表 状态码 状态码英文名称 中文描述 1xx xff08 信息性状态码 xff09 xff1a 请求已被接受 xff0c 需要继续处理 100 Continue 继续 客户端应继续其请求 101 Switching Prot
  • 二进制的加减法_二进制加减法

    二进制的加减法 1 二进制加法 1 Binary Addition Since binary numbers consist of only two digits 0 and 1 so their addition is different
  • SQL注入攻击方法

    SQL注入攻击是一种利用Web应用程序中存在的安全漏洞 xff0c 通过在输入框中插入恶意的SQL代码 xff0c 从而实现对数据库的非法操作 以下是一些常见的SQL注入攻击方法 xff1a 使用单引号 xff08 39 xff09 进行字
  • 利用Python+selenium技术,实现浏览器基本操作详解,代码有详细注释

    首先 xff0c 需要安装selenium库和对应的浏览器驱动程序 以Chrome浏览器为例 xff0c 可以使用以下命令安装selenium和chromedriver xff1a pip install selenium 然后 xff0c
  • &和&&的区别(简单易懂)

    amp xff08 按位与 xff09 和 amp amp xff08 逻辑与 xff09 的区别如下 xff1a 1 amp amp 具有短路功能 xff0c 而 amp 不具有短路功能 2 当 amp 运算符两侧的表达式的结果均为真时
  • Spring框架学习笔记

    一 什么是Spring框架 Spring框架是由于软件开发的复杂性而创建的 Spring使用的是基本的JavaBean来完成以前只可能由EJB完成的事情 然而 xff0c Spring的用途不仅仅限于服务器端的开发 从简单性 可测试性和松耦
  • 人工智能——DBSCAN密度聚类(Python)

    目录 1 概述 1 1 概念 1 2 DBSCAN数据点分类 2 DBSCAN算法流程 2 1 DBSCAN算法流程 xff1a 2 2 举例 3 案例1 xff08 Python实现 xff09 3 1 案例 3 2 Python实现 3
  • @RequestMapping的参数和用法

    在Spring MVC 中使用 64 RequestMapping 来映射请求 xff0c 也就是通过它来指定控制器可以处理哪些URL请求 xff0c 相当于Servlet中在web xml中配置 源码 xff1a 该注解说明可以在类和方法
  • Linux 实验:记录型信号量 生产者-消费者问题详解

    进程同步问题是一个非常重要且相当有趣的问题 xff0c 因而吸引了很多学者对他进行研究 本文就选取其中较为代表性的生产者 消费者问题来进行学习 xff0c 以帮助我们更好的理解进程同步的概念及实现方法 一 问题描述 有一群生产者进程在生产产
  • Linux C语言线程解决生产者与消费者

    前言 生产者 消费者模式 xff0c 生产者这边负责生产产品 而消费者负责消费产品 xff0c 对于消费者来说 xff0c 没有产品的时候只能等待产品出来 xff0c 有产品就使用它 这里我们使用一个变量来表示这个这个产品 xff0c 生产
  • Mariadb

    文章目录 1 数据库的介绍2 mariadb的安装与开启3 软件基本信息4 数据库的安全初始化4 1 执行安全初始化脚本4 2 关闭数据库开放端口 5 数据库的基本管理5 1查看5 2新建5 3更改5 4备份与删除 6 数据库密码管理6 1
  • 【FPGA】按键消抖

    文章目录 一 按键消抖概述1 为何要进行按键消抖2 消抖的方式 二 系统设计1 系统模块划分2 系统时序图 三 代码实现1 按键消抖模块 xff08 key debounce xff09 2 呼吸灯模块 xff08 led breath x
  • java字符串添加字符_如何在Java中向字符串添加字符?

    java字符串添加字符 Case 1 In a simple way we will learn how to add characters by using predefined methods 情况1 xff1a 以简单的方式 xff0
  • 【Ubuntu18.04更换国内源及404错误解决办法】

    Ubuntu18 04 arm换源方法 及 404错误解决办法 换源将下面的任选一组源放入到上面的sources list中 xff0c 96 保存退出并更新即可 96 清华源阿里源中科大源 错误apt get update 后出现 404
  • 【Linux中QT加载.so库与调用Python】

    Linux中QT添加 so库与Python库 一 如何导入 so库1 1 不同系统中 库名称各有不同1 2 Linux中的QT导入库方法 xff1a 二 调用Python2 1 添加Python库2 2 创建Python文件 引入头文件2
  • QT连接SQLServer并添加ODBC数据源

    QT连接SQLServer并添加ODBC数据源 一 创建数据源1 打开ODBC数据源2 创建数据源3 测试数据源 二 QT连接SQLServer1 连接代码2 测试成功样图 一 创建数据源 1 打开ODBC数据源 在搜索框中进行搜索ODBC
  • QT生成exe独立运行文件

    目录 一 封装QT独立运行的 exe文件好处1 1 xff1a 封装软件 xff1a Enigma Virtual Box 9 901 2 xff1a 下载链接 xff1a 阿里云盘 https www aliyundrive com s
  • 【1期 QT之控件的创建与使用】

    前言 QT一开始在1991年被奇趣公司研发 xff0c 创建的目的就是实现GUI图形界面开发与非GUI的开发 后来被诺基亚收购了 xff0c 维护至今 当然在诺基亚手里也是越发展越好 好了QT就介绍这么多了 我们直接上干货 xff1a 我将
  • 【2期 QT信号与槽函数&回调函数与函数指针】

    前言 信号与槽函 xff1a 一对多 多对一 多对多 类似于C 43 43 设计模式中的观察者模式 信号与槽函数不是C 43 43 标准代码 xff0c 是QT特有的 xff0c 最终通过moc meta Object Complier 进
  • 关于约瑟夫环问题的思考(数组做法)

    这几天做题时碰见了一个很有意思的问题 xff0c 也是一个十分经典问题 约瑟夫环问题 问题很简单 xff0c 就是有n个人围成一个圈 xff0c 每隔m个人就自杀一个 xff0c 直到剩下最后一个人为止 xff0c 问最后剩下的最后一个人是