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的更多相关文章

  1. 【新知识】队列&bfs【洛谷p1996约瑟夫问题&洛谷p1451求细胞数量】

    (是时候为五一培训准备真正的技术了qwq) part1  队列(FIFO) 算法简介: FIFO:First In First Out(先进先出) 队列是限定在一端进行插入,另一端进行删除的特殊线性表 ...

  2. 【vector的输出问题】 洛谷 P1996 约瑟夫问题

    题目:P1996 约瑟夫问题 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 可恶啊,本来是一道不算难的题,硬是因为cin,cout同步流卡了我一天qwq 关闭cin,cout同步流 ...

  3. 洛谷——P1996 约瑟夫问题

    P1996 约瑟夫问题 (什么?!要给学弟学妹讲约瑟夫问题?!难道就不怕我给他们讲错了吗?! 啊啊啊,为了不给学弟学妹们讲错,蒟蒻表示要临阵磨一下刀...) 题目背景 约瑟夫是一个无聊的人!!! 题目 ...

  4. 洛谷P1996 约瑟夫问题【链表】

    题目:https://www.luogu.org/problemnew/show/P1996 题意: 约瑟夫环.每次取出第m个,第2m个...... 思路: 链表维护.[感觉很少有用到链表.]非常经典 ...

  5. 洛谷 P1996 约瑟夫问题

    题目背景 约瑟夫是一个无聊的人!!! 题目描述 n个人(n<=100)围成一圈,从第一个人开始报数,数到m的人出列,再由下一个人重新从1开始报数,数到m的人再出圈,……依次类推,直到所有的人都出 ...

  6. (水题)洛谷 - P1996 - 约瑟夫问题 - 链表

    https://www.luogu.org/problemnew/show/P1996 试了一下数组实现的双向链表,是挺难用的,估计是应该写个get_next()函数比直接用next数组好. #inc ...

  7. 洛谷P1996 约瑟夫问题【队列】

    题目背景 约瑟夫是一个无聊的人!!! 题目描述 n个人(n<=100)围成一圈,从第一个人开始报数,数到m的人出列,再由下一个人重新从1开始报数,数到m的人再出圈,--依次类推,直到所有的人都出 ...

  8. 【洛谷P1996】约瑟夫问题

    约瑟夫问题 链表模拟大概是正解 #include<iostream> using namespace std; struct node{ //单链表 int d; node *next; ...

  9. 洛谷P3385负环

    传送门 #include <iostream> #include <cstdio> #include <cstring> #include <algorith ...

随机推荐

  1. chfn是用来改变你的finger讯息

    finger 总览 finger[-lmsp][user...][user host...][[]] 参数: -s Finger显示用户的登录名,真名,终端名以及写状态(如果写被禁止,在终端名后显示一 ...

  2. hdparm命令(转)

    转自:http://man.linuxde.net/hdparm hdparm命令提供了一个命令行的接口用于读取和设置IDE或SCSI硬盘参数. 语法 hdparm(选项)(参数) 选项 -a< ...

  3. SQL中的between and

    "between xx and  xx "在SQL中的用法 这个大家都很熟悉,但是当问到是否包含两端的值时,就有点儿不确定了.在W3School网站上,有这样的一段话: 参考:ht ...

  4. 电子产品使用感受之--Windows 10 小米笔记本Air HDMI转VGA无信号问题

    最近一直通过HDMI转VGA线缆链接我的戴尔P2314H显示器,前天睡觉前,看到电脑上英伟达显卡推了驱动更新,顺手更新了一下,就去睡觉了,转天晚上再用,HDMI接口就没有信号了,上网查了一些信息,获知 ...

  5. MySQL之多表查询一 介绍 二 多表连接查询 三 符合条件连接查询 四 子查询 五 综合练习

    MySQL之多表查询 阅读目录 一 介绍 二 多表连接查询 三 符合条件连接查询 四 子查询 五 综合练习 一 介绍 本节主题 多表连接查询 复合条件连接查询 子查询 首先说一下,我们写项目一般都会建 ...

  6. xdebug的配置

    第一步,让xdebug在php环境中生效 下载xdebug http://www.xdebug.org/download.php 这里会出现针对PHP各种版本的下载.找到适合你自己的版本,此处值得注意 ...

  7. [No000018E]Vim快速跳转任意行、任意列以及高亮显示当前行、当前列方法-Vim使用技巧(3)

    vim提供了丰富的快速跳转任意行.任意列的方法,方便高效地移动光标,定位文件位置. 一.Vim行跳转 使用vim查看文件时,使用以下命令可以快速跳转文件首.尾行,方便对整个文件有个全局把握. 1.1 ...

  8. React 入门实例

    React 入门实例教程 一.安装 React 的安装包,可以到官网下载. $ git clone git@github.com:ruanyf/react-demos.git 如果你没安装 git, ...

  9. OC中如何优化代理是否响应某个方法

    看以下示例代码: if([_delegate respondsToSelector: @selector(someClassDidSomething:)){ [_delegate someClassD ...

  10. Copycat - configure

    Copycat server之间的configure是如何,何时被同步的?   大家可以看到,只有leader可以同步配置   1. 显式的调用LeaderState.configure Leader ...