[洛谷]p1996约瑟夫环 &xdoj1311
https://www.luogu.org/problemnew/show/P1996
约瑟夫环这个问题一直以来都是用循环链表写的,今天才知道有循环队列的写法。以下是要点:
1.循环队列实现环的思想,其实就是队首元素出队,如果它不是该出队的元素,那么就把它继续push进queue,这样就构成了一个环的结构。
2.用一个辅助变量来记录每次要进行的操作,比如每三个人出来选出一个人,temp设置为2,先连续两次出队再入队,此时队首就是该出队的那个元素,把它直接pop扔掉即可。循环执行,直至所有元素出队,打印结果。
#include <bits/stdc++.h>
using namespace std;
int main()
{
int n, m;
scanf("%d %d", &n, &m);
queue<int>Q;
int i;
for (i = ; i <= n; i++)
{
Q.push(i);
}
int temp;
int a;
while (!Q.empty())
{
temp = m-;
while (temp)
{
a = Q.front();
Q.pop();
Q.push(a);
temp--;
}
a = Q.front();
Q.pop();
printf("%d ", a);
}
}
下面再说一下常规的循环链表写法:
1.这个循环链表一定要是双向的链表有prior和next两个指针,如果只有一个指针,删除中间结点的操作无法完成。
2.计数的时候就用简单粗暴的temp辅助变量来完成计数,用cnt从0加到n再置0那种方法容易出错。
#include <bits/stdc++.h>
using namespace std;
typedef struct LNode
{
int data;
struct LNode*prior;
struct LNode*next;
}LNode;
void creat(LNode*&L, int n)
{
LNode*r;
r = L;
LNode*s;
int i;
r->data = ;
for (i = ; i <= n; i++)
{
s = (LNode*)malloc(sizeof(LNode));
s->data = i;
r->next = s;
s->prior = r;
r = r->next;
}
r->next = L;
L->prior = r;
}
void delete1(LNode*&L, int m,int n)
{
int cnt=;
LNode*r;
LNode*s;
r = L;
int temp;
while (n)
{
temp = m-;
while (temp)
{
r = r->next;
temp--;
}
s = r;
printf("%d ", s->data);
r->prior->next = r->next;
r->next->prior = r->prior;
r = r->next;
free(s);
n--;
} }
int main()
{
int n, m;
scanf("%d %d", &n, &m);
LNode *L;
L = (LNode*)malloc(sizeof(LNode));
creat(L, n);
delete1(L,m,n);
}
http://acm.xidian.edu.cn/problem.php?id=1311
下面这道xdoj1311这道题很有意思,仔细想想和约瑟夫队列很相似。
1.第一点就是要想到的是用队列实现环状结构。
2.奇数回合抽出所有2的倍数的编号,偶数回合抽出所有3的倍数的编号。这个实现其实很简单,偶数就是1234每隔两张抽出1张牌,其实就是temp=1进行一次队首移动到队尾的操作,然后现在队首的元素就是该抽出的元素。3同理,移动队首元素两次即可。
3.与约瑟夫环不同的是约瑟夫环是一个循环走到黑直至队空,而这个是很多回合每个回合都要把所有的牌搞一遍,仔细想想就会发现设置t也就是回合次数,可以完成所有牌的遍历,但有一点非常容易错的地方在于,执行完t次操作之后,你的队首元素不再是1(1既不是2的倍数也不是3的倍数始终未被抽出)。要实现队首元素为1就可以等效为每次都是从1开始的,与一条线性的队列没差别了。这里巧妙的用一个while的循环来保证每次队首的元素都是1.这样问题就解决了。
4.最后要实现队列剩余元素的顺序打印就十分简单了,全放在vector中,然后sort排序,输出即可。
整道题目像是一个加强版的约瑟夫,有些细节还是需要注意的。
#include <bits/stdc++.h>
using namespace std;
int main()
{
int T;
scanf("%d", &T);
int n;
queue<int>Q;
while (T--)
{
int cnt = ;
scanf("%d", &n);
int i;
int m;
for (i = ; i <= n; i++)
{
Q.push(i);
}
int temp;
int t;
while (Q.size() > )
{
if (cnt % != )
{
t = Q.size() / ;
while (t--)
{
temp = ;
while (temp--)
{
m = Q.front();
Q.pop();
Q.push(m);
}
Q.pop();
}
while (Q.front() != )
{
m = Q.front();
Q.pop();
Q.push(m);
}
}
else
{
t = Q.size() / ;
while (t--)
{
temp = ;
while (temp--)
{
m = Q.front();
Q.pop();
Q.push(m);
}
Q.pop();
}
while (Q.front() != )
{
m = Q.front();
Q.pop();
Q.push(m);
}
}
cnt++;
}
vector<int>v;
while (!Q.empty())
{
v.push_back(Q.front());
Q.pop();
}
sort(v.begin(), v.end());
for (i = ; i < v.size(); i++)
{
if (i != v.size() - )
printf("%d ", v[i]);
else
printf("%d", v[i]);
}
printf("\n");
}
}
[洛谷]p1996约瑟夫环 &xdoj1311的更多相关文章
- 【新知识】队列&bfs【洛谷p1996约瑟夫问题&洛谷p1451求细胞数量】
(是时候为五一培训准备真正的技术了qwq) part1 队列(FIFO) 算法简介: FIFO:First In First Out(先进先出) 队列是限定在一端进行插入,另一端进行删除的特殊线性表 ...
- 【vector的输出问题】 洛谷 P1996 约瑟夫问题
题目:P1996 约瑟夫问题 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 可恶啊,本来是一道不算难的题,硬是因为cin,cout同步流卡了我一天qwq 关闭cin,cout同步流 ...
- 洛谷——P1996 约瑟夫问题
P1996 约瑟夫问题 (什么?!要给学弟学妹讲约瑟夫问题?!难道就不怕我给他们讲错了吗?! 啊啊啊,为了不给学弟学妹们讲错,蒟蒻表示要临阵磨一下刀...) 题目背景 约瑟夫是一个无聊的人!!! 题目 ...
- 洛谷P1996 约瑟夫问题【链表】
题目:https://www.luogu.org/problemnew/show/P1996 题意: 约瑟夫环.每次取出第m个,第2m个...... 思路: 链表维护.[感觉很少有用到链表.]非常经典 ...
- 洛谷 P1996 约瑟夫问题
题目背景 约瑟夫是一个无聊的人!!! 题目描述 n个人(n<=100)围成一圈,从第一个人开始报数,数到m的人出列,再由下一个人重新从1开始报数,数到m的人再出圈,……依次类推,直到所有的人都出 ...
- (水题)洛谷 - P1996 - 约瑟夫问题 - 链表
https://www.luogu.org/problemnew/show/P1996 试了一下数组实现的双向链表,是挺难用的,估计是应该写个get_next()函数比直接用next数组好. #inc ...
- 洛谷P1996 约瑟夫问题【队列】
题目背景 约瑟夫是一个无聊的人!!! 题目描述 n个人(n<=100)围成一圈,从第一个人开始报数,数到m的人出列,再由下一个人重新从1开始报数,数到m的人再出圈,--依次类推,直到所有的人都出 ...
- 【洛谷P1996】约瑟夫问题
约瑟夫问题 链表模拟大概是正解 #include<iostream> using namespace std; struct node{ //单链表 int d; node *next; ...
- 洛谷P3385负环
传送门 #include <iostream> #include <cstdio> #include <cstring> #include <algorith ...
随机推荐
- chfn是用来改变你的finger讯息
finger 总览 finger[-lmsp][user...][user host...][[]] 参数: -s Finger显示用户的登录名,真名,终端名以及写状态(如果写被禁止,在终端名后显示一 ...
- hdparm命令(转)
转自:http://man.linuxde.net/hdparm hdparm命令提供了一个命令行的接口用于读取和设置IDE或SCSI硬盘参数. 语法 hdparm(选项)(参数) 选项 -a< ...
- SQL中的between and
"between xx and xx "在SQL中的用法 这个大家都很熟悉,但是当问到是否包含两端的值时,就有点儿不确定了.在W3School网站上,有这样的一段话: 参考:ht ...
- 电子产品使用感受之--Windows 10 小米笔记本Air HDMI转VGA无信号问题
最近一直通过HDMI转VGA线缆链接我的戴尔P2314H显示器,前天睡觉前,看到电脑上英伟达显卡推了驱动更新,顺手更新了一下,就去睡觉了,转天晚上再用,HDMI接口就没有信号了,上网查了一些信息,获知 ...
- MySQL之多表查询一 介绍 二 多表连接查询 三 符合条件连接查询 四 子查询 五 综合练习
MySQL之多表查询 阅读目录 一 介绍 二 多表连接查询 三 符合条件连接查询 四 子查询 五 综合练习 一 介绍 本节主题 多表连接查询 复合条件连接查询 子查询 首先说一下,我们写项目一般都会建 ...
- xdebug的配置
第一步,让xdebug在php环境中生效 下载xdebug http://www.xdebug.org/download.php 这里会出现针对PHP各种版本的下载.找到适合你自己的版本,此处值得注意 ...
- [No000018E]Vim快速跳转任意行、任意列以及高亮显示当前行、当前列方法-Vim使用技巧(3)
vim提供了丰富的快速跳转任意行.任意列的方法,方便高效地移动光标,定位文件位置. 一.Vim行跳转 使用vim查看文件时,使用以下命令可以快速跳转文件首.尾行,方便对整个文件有个全局把握. 1.1 ...
- React 入门实例
React 入门实例教程 一.安装 React 的安装包,可以到官网下载. $ git clone git@github.com:ruanyf/react-demos.git 如果你没安装 git, ...
- OC中如何优化代理是否响应某个方法
看以下示例代码: if([_delegate respondsToSelector: @selector(someClassDidSomething:)){ [_delegate someClassD ...
- Copycat - configure
Copycat server之间的configure是如何,何时被同步的? 大家可以看到,只有leader可以同步配置 1. 显式的调用LeaderState.configure Leader ...