使用STL库list类实现单双向约瑟夫环问题(C++)

2023-05-16

目录

一、单向约瑟夫环 

1.问题描述

2.list类函数用法  

(1)list构造

(2)list iterator迭代器

(3)list容量

(4)list元素访问

(5)list增删查改

(6)list算法操作

3.代码演示

4.结果演示 

 

二、双向约瑟夫环

1.问题描述

2.代码演示

3.结果演示


一、单向约瑟夫环 

1.问题描述

       约瑟夫生者死者游戏的大意是:30个旅客同乘一条船,因为严重超载,加上风高浪大,危险万分;因此船长告诉乘客,只有将全船一半的旅客投入海中,其余人才能幸免遇难。无奈,大家只得同意这种办法,并议定30个人围成一圈,由第一个人开始,依次报数,数到第9人,便把他投入大海中,然后从他的下一个人数起,数到第9人,再将他投入大海,如此循环,直到剩下15个乘客为止。问哪些位置是将被扔下大海的位置。

       为了解决这一问题,可以用长度为30的数组作为线性存储结构,并把该数组看成是一个首尾相接的环形结构,那么每投入大海一个乘客,就要在该数组的相应位置做一个删除标记,该单元以后就不再作为计数单元。这样做不仅算法较为复杂,而且效率低,还要移动大量的元素。用单循环链表解决这一问题,实现的方法相对要简单得多。首先要定义链表结点,单循环链表的结点结构与一般的结点结构完全相同,只是数据域用一个整数来表示位置;然后将它们组成具有30个结点的单循环链表。接下来从位置为1的结点开始数,数到第8个结点,就将下一个结点从循环链表中删去,然后再从删去结点的下一个结点开始数起,数到第8个结点,再将其下一个结点删去,如此进行下去,直到剩下15个结点为止。

       为了不失一般性,将30改为一个任意输入的正整数n,而报数上限(原为9)也为一个任选的正整数m。并且我们知道C++中有STL库可以调用list类进行链表的各种操作,所以代码就能变得很精炼。下面先来介绍一下list类的函数用法。

 

2.list类函数用法  

(1)list构造

构造函数 接口说明
list() 构造空的list
list (size_type n, const value_type& val = value_type())构造的list中包含n个值为val的元素
list (const list& x) 拷贝构造函数
list (InputIterator first, InputIterator last) 

用[first, last)区间中的元素构造list

 

(2)list iterator迭代器

   这里的迭代器理解成一个指针,该指针指向list中的某个节点。

函数声明接口类型
begin()返回第一个元素的迭代器
end()返回最后一个元素下一个元素的迭代器
rbegin()返回第一个元素的reverse_iterator,即end位置
rend()返回最后一个元素下一个位置的reverse_iterator,即begin位置

(3)list容量

函数声明接口说明
empty()检测list是否为空,是返回true,否则返回false
size()返回list中有效节点的个数

(4)list元素访问

函数申明接口说明
front()返回list的第一个节点中值的引用
back()返回list最后一个节点中值的引用

(5)list增删查改

函数声明 接口说明
push_back() 尾插
push_front() 头插
pop_back()尾删
pop_front()头删
insert()  在pos位置插入值为val的元素
erase() 删除pos位置的元素
swap()交换两个元素
clear() 清空list中的有效元素

(6)list算法操作

函数声明接口说明
remove()删除指定值的元素
reverse()反转链表
unique()去重
merge()合并两个有序链表
sort()链表排序

   

3.代码演示

#include<iostream>
#include<list>      //list容器的头文件
using namespace std;

int main()
{	
	int m, n, count=1;//计数器 
	
	cout<<"请输入总人数n\n";
	cin >> n;
	cout<<"请输入数到m退出的数m\n";
	cin >> m;
	
	list <int> l;//创建链表
	for(int i=1;i<=n;i++)
	{
		l.push_back(i);
	}//对n个元素标号
	
	list<int>::iterator iter = l.begin();  //设置一个迭代器指向头部
	
	cout << "船上的人的编号为:" << endl;
	
	for(iter = l.begin(); iter != l.end(); iter++)
	{
		cout << *iter << " ";
	}
	cout << endl;
		
	iter = l.begin();   //让迭代器指向第一个元素
	while(l.size() > (n/2))//链表人数如果等于一半人,那么删除操作结束
	{
		for(int i = 0 ; i < m-1; i++) //迭代器位移m-1次,表示数到m 
		{
			if(iter == --l.end())//如果迭代器指向了链表最后一个元素,即l.end()的前一位,则返回到第一个节点处 
			{
				iter = l.begin();
			}
			else
			{ 
				iter++;//如果迭代器没有指向最后一个元素,那么就正常自增
			}
		}
		
		cout << "第" << count << "次离开的人的序号是" << *iter << endl;
			
		if(iter != --l.end())      //如果不是最后一个元素,正常删除即可 
		{
			iter = l.erase(iter); //erase()函数返回的是删除元素下一位,这里是防止下一位是end()的情况
		}
		else
	   	{
	   		l.pop_back();  //如果是最后一个元素的话,直接调用pop_back函数尾删,并且迭代器直接返回到链表第一个节点 
			iter = l.begin();
		}	
			
		
		count++;
	}
		
		cout << "船上还剩下的人为:" << endl;
		
		for(iter = l.begin(); iter != l.end(); iter++)
		{
			cout << *iter << " ";
		}
		cout << endl;
		
		
	return 0;
 } 


4.结果演示 

 


二、双向约瑟夫环

1.问题描述

       约瑟夫双向生死游戏是在约瑟夫生者死者游戏的基础上,正向计数后反向计数,然后再正向计数。具体描述如下:30个旅客同乘一条船,因为严重超载,加上风高浪大,危险万分;因此船长告诉乘客,只有将全船一半的旅客投入海中,其余人才能幸免遇难。无奈,大家只得同意这种办法,并议定30个人围成一圈,由第一个人开始,顺时针依次报数,数到第9人,便把他投入大海中,然后从他的下一个人数起,逆时针数到第5人,将他投入大海,然后从他逆时针的下一个人数起,顺时针数到第9人,再将他投入大海,如此循环,直到剩下15个乘客为止。问哪些位置是将被扔下大海的位置。

       本游戏的数学建模如下:假设n个旅客排成一个环形,依次顺序编号1,2,…,n。从某个指定的第1号开始,沿环计数,数到第m个人就让其出列,然后从第m+1个人反向计数到m-k+1个人,让其出列,然后从m-k个人开始重新正向沿环计数,再数m个人后让其出列,然后再反向数k 个人后让其出列。这个过程一直进行到剩下q个旅客为止。

本游戏的要求用户输入的内容包括:

1. 旅客的个数,也就是n的值;

2. 正向离开旅客的间隔数,也就是m的值;

3. 反向离开旅客的间隔数,也就是k的值;

4. 所有旅客的序号作为一组数据要求存放在某种数据结构中。

本游戏要求输出的内容是包括

1. 离开旅客的序号;

2. 剩余旅客的序号;

所以,根据上面的模型分析及输入输出参数分析,可以定义一种数据结构后进行算法实现。

       约瑟夫双向生死游戏如果用单循环链表作为线性存储结构,就只能正向计数结点,反向计数比较困难,算法较为复杂,而且效率低。用双向循环链表解决这一问题,实现的方法相对要简单得多。

为了不失一般性,将30改为一个任意输入的正整数n,而正向报数上限(原为9)也为一个任选的正整数m,正向报数上限(原为5)也为一个任选的正整数k。

 

 

2.代码演示

#include<iostream>
#include<list>//list容器的头文件
using namespace std;
int main()
{	
	int m, n, k, count=1;//计数器 
	
	cout<<"请输入总人数n\n";
	cin >> n;
	cout<<"请输入顺时针报数的数m:\n";
	cin >> m;
	cout<<"请输入逆时针报数的数k:\n";
	cin >> k;
	
	list <int> l;//创建链表
	for(int i=1;i<=n;i++)
	{
		l.push_back(i);
	}//对n个元素标号
	
	list<int>::iterator iter = l.begin();  //设置一个迭代器指向头部

	
	cout << "船上的人的序号为:" << endl;
	
	for(iter = l.begin(); iter != l.end(); iter++)
	{
		cout << *iter << " ";
	}
	cout << endl;
	
	iter = l.begin();      //让迭代器指向第一个元素 
	while(l.size()  >  (n / 2))//链表人数如果等于一半人,那么删除操作结束
	{
		for(int i = 0 ; i < m-1; i++) //迭代器向后位移m-1次,表示数到m 
		{
			if(iter == --l.end())//如果迭代器指向了链表最后一个元素,即l.end()的前一位,则返回到第一个节点处 
			{
				iter = l.begin();
			}
			else
			{ 
				iter++;//如果迭代器没有指向最后一个元素,那么就正常自增
			}
		}
		iter++;  //顺时针数完从他的后一个开始逆时针报数 
		
		for(int i = 0 ; i < k-1; i++) //迭代器向前位移k-1次,表示数到k 
		{
			if(iter == l.begin())//如果迭代器指向了链表的头结点,即l.begin,则返回到尾节点处 
			{
				iter = --l.end();
			}
			else
			{ 
				iter--;//如果迭代器没有指向头结点,那么就正常自减 
			}
		}
		
		cout << "第" << count << "次离开的人的序号是" << *iter << endl;
			
		if(iter != --l.end())              //如果不是最后一个元素,正常删除即可 
		{
			iter = l.erase(iter);        //erase()函数返回的是删除元素下一位,这里是防止下一位是end()的情况
			iter--;						//(删除完后下一轮要从前一个元素开始正向报数)
		}
		else
	   	{
	   		l.pop_back();        //如果是最后一个元素的话,直接调用pop_back函数尾删,并且迭代器直接返回到链表尾节点
			iter = l.end();      //(删除完后下一轮要从前一个元素开始正向报数) 	
		}	
		
		count++;
	}
		
		cout << "船上还剩下的人为:" << endl;
		
		for(iter = l.begin(); iter != l.end(); iter++)
		{
			cout << *iter << " ";
		}
		cout << endl;
	
	
	return 0;
 } 

3.结果演示

 

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

使用STL库list类实现单双向约瑟夫环问题(C++) 的相关文章

  • c/c++输入带空格、tap、换行符的字符串

    1 scanf charstr 10 scanf 34 s 34 str 123 adw 其实只输入了 123 1 不读入空格和回车还有tap键 从空格处结束 2 输入字符串长度超过字符数组元素个数不报错 xff0c 只是不读入 3 当输入
  • 如何防止网站被爬虫爬取的几种办法

    相较于爬虫技术 xff0c 反爬虫实际上更复杂 目前许多互联网企业都会花大力气进行 反爬虫 xff0c 网络爬虫不但会占据过多的网站流量 xff0c 导致有真正需求的用户没法进入网站 xff0c 另外也有可能会导致网站关键数据的外泄等现象
  • 指定位置插入字符串(c++insert函数、find函数使用)

    一 insert函数 xff08 插入函数 xff09 str1 61 str1 xff08 被插入字符串 xff09 insert 插入位置 str2 xff08 被插入字符串 xff09 xff0c n xff0c m ps xff1a
  • vector容器(C++黑马程序员笔记)

    3 2 vector容器 3 2 1 vector基本概念 功能 vector数据结构和数组非常相似 xff0c 也称为单端数组 vector与普通数组区别 不同之处在于数组是静态空间 xff0c 而vector可以动态扩展 动态扩展 并不
  • set容器(c++黑马程序员笔记)

    3 8 set multiset容器 3 8 1 set基本概念 简介 所有元素都会在插入时自动被排序 本质 set multiset属于关联式容器 xff0c 底层结构是用二叉树实现 set和multiset区别 set不允许容器中有重复
  • map函数

    3 9 map multimap容器 3 9 1 map基本概念 简介 map中所有元素都是pair pair中第一 个元素为key 键值 xff0c 起到索引作用 xff0c 第二个元素为value 实值 所有元素都会根据元素的键值自动排
  • 解决Maven配置本地仓库路径不生效问题多个方法详解。(已成功解决自己遇到的问题)

    首先我尝试了很多种方法 xff0c 就是这个方法让我成功 xff0c 和大家分享一下 xff01 xff08 我用方法二成功的 xff01 xff09 maven本地仓库默认值 xff1a 用户家目录 m2 repository 由于本地仓
  • Servlet快速学习和Tomcat快速部署(web)

    Servlet快速学习和Tomcat快速部署 xff08 web xff09 一 快速入门二 执行流程三 生命周期四 Servlet体系结构五 Servlet urlPattern配置六 XML配置方式编写Servlet七 Tomcat快速
  • 蓝桥杯必备模板(python)

    蓝桥杯必备算法模板 xff08 python xff09 xff1a 前缀和模板差分模板二分双指针位运算最大公约数和最小公倍数模板判断质数和埃氏筛法模板唯一分解定理和质因数分解关系和模板快速幂并查集区间合并回溯算法模板 xff1a DFS
  • 蓝桥杯必备模块及常用操作(python)

    蓝桥杯必会模块 xff08 python xff09 xff1a 字符类型模块日期函数模块 常用 优先级队列itertools模块collections模块Bisect模块List 集合set 集合Math模块 字符类型模块 先看点常用但比
  • string的几个常见库函数

    文章目录 前言一 strlen直接求字符串长度二 strcpy用来赋值三 strcat追加字符串四 strcmp用于比较两个字符串五 strstr子串查找六 strtok应用七 加长度限制八 strerror 九 其他操作1 iscntrl
  • vector详解

    在开始学习C 43 43 的STL之后 xff0c 相信大家都学会通过查文档来了解一些库函数 xff0c 今天我来给大家介绍vector xff0c 从基本的使用到vector背后的源码实现 xff0c 迭代器等展开讲 xff0c 仔细阅读
  • 如何利用开源思想开发一个SEO友好型网站

    当你对一个网站进行 SEO 优化的时候 xff0c 不要期望你的努力能立即得到回报 耐心等待并更正内容营销策略 xff0c 最终会发现你的网站很受用户欢迎 下面就教你如何利用开源思维开发一个SEO友好型网站 xff01 首先 xff0c 你
  • 【STM32】串口通讯USART串口中断配置

    目录 STM32 USART 简介 程序编写 硬件接线 实际波形 STM32 USART 简介 STM32的USART通用同步异步收发器是一个串行通信设备 xff0c 可以灵活的与外部设备进行全双工数据交换 有别于USART xff0c 还
  • openmv第三天之标记跟踪

    AprilTag是一个视觉基准系统 感觉就是一个可以定位 校准帮助openmv来找到 定位的东西 官方解释的用处 xff1a 简单来说 xff0c 只要把这个tag贴到目标上 xff0c 就可以在OpenMV上识别出这个标签的3D位置 xf
  • 【A_star三维路径规划】A_star算法无人机三维路径规划【含Matlab源码 1387期】

    一 A star算法简介 0 引言 随着现代技术的发展 飞行器种类不断变多 应用也日趋专一化 完善化 如专门用作植保的大疆PS X625无人机 用作街景拍摄与监控巡察的宝鸡行翼航空科技的X8无人机 以及用作水下救援的白鲨MIX水下无人机等
  • 【路径规划】A_star算法机器人动态避障路径规划【含Matlab源码 1033期】

    一 A star算法简介 1 A Star算法及其应用现状 进行搜索任务时提取的有助于简化搜索过程的信息被称为启发信息 启发信息经过文字提炼和公式化后转变为启发函数 启发函数可以表示自起始顶点至目标顶点间的估算距离 也可以表示自起始顶点至目
  • C语言打印输出字符串的几种方法

    思路分析 知识点补充 1 xff0c 在C语言中 xff0c 一维数组的数组名实际上就是指向数组首项元素的指针 2 xff0c 如果指针p已经指向一个字符串 xff0c 判断字符串是否结束 xff0c 一般采用while p 61 39 0
  • 对‘fmt::v9::detail::throw_format_error(char const*)’未定义的引用

    在学习视觉SLAM十四讲第十二章的过程中 xff0c 尝试跑了一下单目稠密地图构建的代码 xff0c 在编译源代码的过程中遇到该问题 xff0c 编译结果中显示长篇的 CMakeFiles dense mapping dir dense m
  • [STM32F103C8T6] 串口

    1 非中断串口发送数据 串口发送 接收函数 xff1a HAL UART Transmit 串口发送数据 xff0c 使用超时管理机制 HAL UART Receive 串口接收数据 xff0c 使用超时管理机制 HAL UART Tran

随机推荐

  • 【C语言】strcat函数_字符串追加/连接

    前言 xff1a 在C C 43 43 的学习过程当中一定一定要多刷题 xff0c 牛客网作为国内内容超级丰富的IT题库 xff0c 尤其是它的C C 43 43 xff0c 有从入门到大厂真题 xff0c 而且大部分的考试题目也是从中抽取
  • SWUST OJ448: 字符串查找

    题目描述 在一段句子中找出给定字符串出现在句子中第一个字母出现的位置 句子中字符个数小于4500 字符串字符个数小于120 输入 两行 第一行是给定字符串 第二行是句子 输出 整数 xff0c 字符串出现的位置 样例输入 abcde thi
  • C语言十进制转十六进制

    输入 xff1a 123 输出 xff1a 7B include lt stdio h gt int main int n scanf 34 d 34 amp n int a 100 int count int i 61 0 while 1
  • CentOS软件那么老为什么大家还要用它?

    作为一个专业的服务器系统 xff0c RHEL 系统理论上每一个软件包都有 RedHat 内部的人员负责维护 xff0c 这个维护包括长期 xff08 和系统生命周期一样长 xff09 的开发 更新 测试 运维等 也就是说你能从 RHEL
  • vscode配置C/C++调试环境

    转自作者知乎的原创文章 vscode配置C 43 43 调试环境 在环境变量path中添加mingw中bin后 在vscode界面侧边栏点击调试界面 xff0c 创建launch json文件 添加配置后保存就行 下为我的添加配置后自动生成
  • ROS通信机制~话题通信(Publisher&Subscriber)·笔记2

    系列文章目录 xff1a ROS开发 xff08 ubuntu xff09 笔记 1 嘻 嘻的博客 CSDN博客 ROS通信机制 服务通信 server amp client 笔记3 嘻 嘻的博客 CSDN博客 话题通信 理论模型 xff1
  • postman Windows的详细安装过程

    1 首先去postman官网下载postman 官网地址 xff1a Download Postman Get Started for Free https www postman com downloads 2 根据你浏览器下载的所在位置
  • 数据结构链表的结构体定义的理解(C语言)

    此理解有参考数据结构的教材和一些博主的文章 xff1b 1 先补充一下typedef在此处的基本用法 xff1a typedef可用来建立已经定义好的数据类型的别名 形式为 xff1a typedef 已有变量名 别名 通俗的来说就是给已有
  • ROS安装与Rviz的摄像头视频采集与标定

    文章目录 一 ROS的安装与配置1 添加 ROS 软件源 xff0c 将下列命令输入到 Ubuntu 的终端执行2 添加密钥 xff0c 将下列命令输入到 Ubuntu 的终端执行3 安装desktop full4 初始化rostep5 设
  • HTTP协议的基本格式

    目录 一 HTTP请求 1 1 首行 1 1 1 URL 1 1 2 方法 1 2 请求报头 xff08 header xff09 1 2 1 host 编辑 1 2 2 Content Length和Content Type 1 2 3
  • 思岚雷达win与ubuntu18.04连接并测试详细过程

    雷达简介 包含套件 雷达模组 xff08 内置pwm电机驱动 xff09 usb适配器 Micro USB线缆 电源线 接线方式 ps 雷达不需额外的电源供电 xff0c 直接使用电脑USB接口 xff0c 5V供电 驱动安装 USB 适配
  • c/c++常见字符串函数(strlen,strcmp,strcat,strcpy,strstr,strncpy,strncat,strncmp)的详解和自己编辑实现

    下面介绍c语言中指针与数组的面试题 再看下列题之前首先要知识储备 下面用c语言介绍常用字符串函数和进行编译 一 介绍strlen函数 unsigned int strlen char s 或size t strlen const char
  • 【C语言】学数据结构前必学的结构体struct详细

    佛祖说 xff0c 他可以满足程序猿一个愿望 程序猿许愿有生之年写出一个没有bug的程序 xff0c 然后他得到了永生 目录 1 结构体的声明与定义 1 1结构体是什么 xff1f 1 2为什么要有结构 xff1f 1 3结构体的声明 1
  • 数据结构哈希查找的C语言实现

    大家好 xff0c 我是练习编程时长两年半的昆工第一ikun xff0c 今天我们来分享查找算法中的一个 哈希查找 xff0c 哈希查找适用于有庞大的数据量时的查找 xff0c 是一种很好用的查找算法 xff0c 话不多说 xff0c 开团
  • 为什么很多程序员喜欢linux系统?

    a gt Linux哪些行业在运用 xff1f Linux系统运用极其广泛 xff0c 不少用户只知道windows xff0c 是因为 xff0c Linux的运用主要是在企业端 现在科技极其发达 xff0c 我们手机在手 xff0c 就
  • Linux Ubuntu下的标准IO相关库函数的介绍与使用

    大家好 xff0c 我是练习编程时长两年半的个人练习生昆工第一ikun xff0c 今天我们来分享一下标准IO相关函数库的介绍与使用 xff0c 话不多话 xff0c 开团 xff01 xff01 xff01 xff01 xff01 目录
  • Linux Ubuntu下的文件IO介绍及实例应用(C语言)

    大家好 xff0c 我是练习编程时长两年半的个人练习生昆工第一ikun xff0c 昨天咋们说了标准IO xff0c 今天咋们来分享文件IO xff0c 以及一个很有趣的实例 xff0c 给图片加密 xff0c 使其无法打开 话不多说 xf
  • 输入年月日得出该天是星期几(C语言)

    大家好 xff0c 我是练习编程时长两年半的个人练习生昆工第一ikun xff0c 昨天因为在写Thoughtworkers的2018年笔试题 xff0c 所以没有更新 xff0c 今天就先把笔试题中的一个函数分享出来 xff0c 该函数可
  • Linux下的UDP服务器客户端的搭建(C语言实现)

    大家好 xff0c 我是练习编程时长两年半的个人练习生昆工第一ikun xff0c 昨天我们说了搭建TCP的服务器和客户端 xff0c 今天我们就来分享一下UDP的服务器和客户端搭建 UDP的特点是无连接 xff0c 多个客户端可以发送消息
  • 使用STL库list类实现单双向约瑟夫环问题(C++)

    目录 一 单向约瑟夫环 1 问题描述 2 list类函数用法 xff08 1 xff09 list构造 xff08 2 xff09 list iterator迭代器 xff08 3 xff09 list容量 xff08 4 xff09 li