Made by 退役的OIer

第一次写博客,写得不好 or 不清楚的可以 在下方留言,我会尽量改进的!


好啦~~~回到正题,题面见传送门

【问题描述】

  采药人最近在认真切题,但回旋的转盘时常在眼前浮现,旧日的美好时光总把他带回那个无忧无虑的押忍战斗应援团的时代。于是他在切题的时候也按照这样的规则以怀念旧时光。
  规则是这样的:采药人手头有 n 道题,他会将其按 1 到 n 的顺序绕成一个环,他会从 1开始按照 1 到 n 顺序每隔 p 道没写过的写一题,直到这些题都写完了。他想知道这些题中第 k 个被他切掉的题是第几题?
  如有 7 道题,采药人每隔 3 道写一题。那么他写题的顺序是 3 6 2 7 5 1 4。

【输入格式】

第一行,一个整数 t,表示数据组数
接下来 t 行,每行三个正整数 n,p,k,意义如题目所述

【输出格式】

t 行,每行一个整数,表示答案

【输入输出样例】

rule.in
1
7 3 7
rule.out
4

【数据规模】

对于 20%,k<=n<=1000,p<=100
对于 40%,k<=n<=10000,p<=1000
对于另外 20%,k<=n<=1000000000,p<=2
对于 100%,t<=10,k<=n<=10^14,p<=10000

solution:

  首先我们可以看出这就是Joseph问题(如果不会请移步至Joseph问题

最最基础的算法就是模拟啦~~~

数p道题就切掉当前这个,重复k次,我们当然可以使用循环链表模拟这个过程(很形象的)

部分代码如下:

void pre()
{
FOR(i,,n)
{
l[i]=(i==)?n:i-;
r[i]=(i==n)?:i+;
}
return;
}
void del(int x)//在链表中删除x(即把x两边的连起来)
{
r[l[x]]=r[x];
l[r[x]]=l[x];
return;
}
//在main中
int pos=n;
while(k-->)
{
FOR(i,,p) pos=r[pos];
del(pos);
}
printf("%d",pos);

哈哈,是不是很简单?

然而,仔细一想,整个算法时间复杂度为O(pk)

emmm....显然炸得裂开

我们急需寻找一个更为高效的算法

递推!

我们发现,一个队列进行一次切题操作后,长度会减小1,而队列开头到切掉的题的下一个

即可以从此处重新编号1~(n-1),相当于对当前队列编号整体减p

(自己推一推,模拟  切 一 道 题  的过程)

假设f[i][j]为队列长度为 i 时 , 切掉的第 j 道题的编号

则有以下递推式:

初态即

又可以发现,f数组没有必要,只用一个变量pos即可

那么代码就出来了:

//在main中
int pos=(p-)%(n-k+)+;
FOR(i,n-k+,n)
{
pos=(pos+p-)%i+;
}
printf("%d",pos);

分析一波发现,算法时间复杂度降到了O(k)

wow~~~优(yiu)秀(xiu)

欣喜地看看数据范围, ?@#(*%&@#!

k<=n<=10^14

这说明我们需要更~~~~~~~~高效的算法!

???(仔细想想怎么优化,想完再往下看)




就一个小优化即可:乘法加速

用乘法加速加法运算(是不是很简单)

(不对呀,这里有取模,怎么加速???)

很容易发现,取模并不是每次都有改变,就是说取模把加法分成了若干小段,每个小段中pos是连加的

每次 i 增加 1 时, pos 增加 p(是不是很像追及问题?)

那么每段中, i 就会增加 tmp, 其中

这样就很好办了~~~

再就不多说了,见完整代码

#include<cstdio>
#define ll long long
#define rll register ll
ll n,p,k,len,pos;
int T;
int main()
{
scanf("%d",&T);
while(T-->)
{
scanf("%lld%lld%lld",&n,&p,&k);
if(p==) printf("%lld\n",k);//注意特判p==1的时候
else
{
len=n-k+;pos=(p-)%len+;
for(rll tmp();len+tmp<=n;tmp=(len-pos)/(p-)+)
pos=(pos+tmp*p-)%(len+=tmp)+;
printf("%lld\n",(pos+(n-len)*p-)%n+);
}
}
return ;
}

是不是很简单?

复杂度也很简单,O(T * F(k) )(其中F(k)为关于k的函数,值很小,可近似为log(k))

愉快地  切 一 道 题  啦~~~


最后,

感谢   P.Y.Y  提供的官方题解平台~~~

感谢各位   巨佬&&神犇   的支持与鼓励~~~

可别忘了指正,转发与推荐哟~

(链接:https://www.cnblogs.com/HSY-2019/p/12367741.html)

【题解】 2月19日 厦门双十中学NOIP2014模拟D2 T1 采药人的切题规则的更多相关文章

  1. 【题解】 2月19日 厦门双十中学NOIP2014模拟D2 T2 采药人接水果

    [问题描述] 采药人虽然 AFO(SU),但他在闲暇的时候还是可以玩一玩接水果(cat)的.但他渐渐发现 cat 好像有点太弱智.于是他不想浪费他的智商,于是决定写一个程序帮他玩. cat 是这样玩的 ...

  2. 【Usaco 2009 Gold】JZOJ2020年9月19日提高B组T4 过路费

    [Usaco 2009 Gold]JZOJ2020年9月19日提高B组T4 过路费 题目 Description 跟所有人一样,农夫约翰以着宁教我负天下牛,休叫天下牛负我的伟大精神,日日夜夜苦思生财之 ...

  3. 【Usaco 2009 Gold】JZOJ2020年9月19日提高B组T3 头晕的奶牛

    [Usaco 2009 Gold]JZOJ2020年9月19日提高B组T3 头晕的奶牛 题目 Description 奶牛们发现,在农场里面赛跑是很有趣的一件事.可是她们一旦在农场里面不断地转圈,就会 ...

  4. 【Usaco 2009 Gold 】JZOJ2020年9月19日提高B组T2 电视游戏问题

    [Usaco 2009 Gold ]JZOJ2020年9月19日提高B组T2 电视游戏问题 题目 Description 农夫约翰的奶牛们游戏成瘾!本来FJ是想要按照陶叫兽的做法拿她们去电击戒瘾的,可 ...

  5. 【Usaco 2009 Silver】JZOJ2020年9月19日提高B组T1 音乐节拍

    [Usaco 2009 Silver]JZOJ2020年9月19日提高B组T1 音乐节拍 题目 Description FJ准备教他的奶牛弹奏一首歌曲,歌曲由N(1<=N<=50,000) ...

  6. Python array,list,dataframe索引切片操作 2016年07月19日——智浪文档

    array,list,dataframe索引切片操作 2016年07月19日——智浪文档 list,一维,二维array,datafrme,loc.iloc.ix的简单探讨 Numpy数组的索引和切片 ...

  7. 2016年12月19日 星期一 --出埃及记 Exodus 21:14

    2016年12月19日 星期一 --出埃及记 Exodus 21:14 But if a man schemes and kills another man deliberately, take hi ...

  8. 2016年11月19日 星期六 --出埃及记 Exodus 20:10

    2016年11月19日 星期六 --出埃及记 Exodus 20:10 but the seventh day is a Sabbath to the LORD your God. On it you ...

  9. 2016年10月19日 星期三 --出埃及记 Exodus 19:3

    2016年10月19日 星期三 --出埃及记 Exodus 19:3 Then Moses went up to God, and the LORD called to him from the mo ...

随机推荐

  1. 在动作方法中生成输出URL (Generating Outgoing URLs in Action Methods) |

  2. allegro使用经验总结(一)

    在用allegro开发flappy bird.游戏虽然小,但是用到了allegro的方方面面,可以说是"麻雀虽小五脏俱全". 1.physfs 这是一个跨平台的读写文件的库,可以直 ...

  3. 工具之sed

    转自:http://www.cnblogs.com/dong008259/archive/2011/12/07/2279897.html sed是一个很好的文件处理工具,本身是一个管道命令,主要是以行 ...

  4. 数据结构与算法 --- js描述栈

    js描述栈及栈的使用 栈的特性就是只能通过一端访问,这一段就是叫做栈顶.咖啡馆内的一摞盘子就是最形象的栈的例子: 根据栈的特性,就可以定义栈的一些特殊属性和方法;用js的描述栈的时候底层数据结构用的是 ...

  5. CSRF攻击原理

    CSRF CSRF(Cross-site request forgery)跨站请求伪造,CSRF是一种夹持用户在已经登陆的web应用程序上执行非本意的操作的攻击方式.相比于XSS,CSRF是利用了系统 ...

  6. centos7中修改运行级别

    centos6 在centos6里打开vim /etc/inittab文件看到下面有一行 id:5:initdefault,因此我们可以通过修改这个文件的id后的数字来修改运行级别 如果我们想要直接切 ...

  7. flask中使用SQLAlchemy操作mysql的一些注意事项和坑

    一 ImportError: cannot import name 'db' 由于app最后才加载,所以其他文件,比如models.py不能从app.py导入任何变量, 要使用db可以先定义一个,之后 ...

  8. JVM基础快速入门篇

    Java是一门可以跨平台的语言,但是Java本身是不可以实现跨平台的,需要JVM实现跨平台.javac编译好后的class文件,在Windows.Linux.Mac等系统上,只要该系统安装对应的Jav ...

  9. 面试官:"谈谈分库分表吧?"

    转自:学习Java的小姐姐 www.cnblogs.com/chenchen0618/p/11624480.html 1.什么是分库分表 从字面上简单理解,就是将原本存储在一个库的数据分块存储在多个库 ...

  10. windows下使用pycharm配置python的virtualenv环境

    环境:win10 python2.7.10(64) 在path中配置python环境 注意:我这边的环境是py2和py3共存的,但是我这边默认使用的是py2. 在virtualenv环境中安装指定的p ...