[POI2011]SEJ-Strongbox
题目大意:
一个有密码箱,数字是0~n-1,其中有若干个密码,密码的特点:若x是密码,y是密码,(x可以等于y)则(x+y)%n也是密码。
给一个n(<=10^14),一个k(k<=min(250000,n)),给k个数(a[k]<n),前k-1个数不是密码,第k个数是密码。
求在0~n-1中,最多有多少个数字是密码?
题解:
看起来和数论有一些关系。
而且一定是一个性质题。
结论1:若x是密码,则gcd(n,x)是密码
发现,x是密码,则k*x%n都是密码。
所以,一定存在一个t,c,使得t*x-n*c=gcd(n,x)
并且根据裴属定理,不能用x凑出一个更小的密码比gcd(n,x)更小,
结论2:若x,y是密码,则gcd(x,y)是密码。
根据裴属定理,p*x+q*y=gcd(x,y)有整数解。
如果q是负数q=-q,那么就是p*x+(c*n-q)*y=gcd(x,y)+c*n*y
那么,就存在非负数p,q使得p*x+q*y=gcd(x,y) mod n
结论3:若x是所有密码中最小的那一个,那么,所有的密码就是x,2x,3x,...kx,并且x是n的约数。
反证。设x是最小的,y是另一个密码,若x不是y的约数,那么gcd(x,y)<x,根据结论二,那么gcd(x,y)就是一个更小的密码。矛盾。
所以,任意的y都是x的倍数。
由于对于一个密码z,根据结论1,gcd(n,z)也是密码,所以,最小的密码x是gcd(n,z)的约数,也就是n的约数。
所以,如果我们求出了满足条件的x,那么n/x就是答案。
我们密码数量最多,所以,x必须取最小的。
由于给了一个a[k]是密码,而x又是n的约数,所以x就一定是gcd(a[k],n)的约数。
并且,x不能是a[1~k-1]的约数,只要是,那么x就能凑出ai,与ai不是密码矛盾。
只要不存在这样的ai,那么x一定可以是最小的密码(裴属定理可以证明)。
所以,我们可以枚举gcd(a[k],n)的约数,然后排序。
从小到大枚举x,再暴力验证是否是a[i]的约数,第一个符合的x,n/x就是答案。
O(ksqrt(n))
代码:
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=+;
ll n,k;
ll gcd(ll a,ll b){
return b?gcd(b,a%b):a;
}
ll a[N],fac[N];
int tot;
int main(){
scanf("%lld%lld",&n,&k);
for(int i=;i<=k;i++)scanf("%lld",&a[i]);
ll g=gcd(a[k],n);
//cout<<"gg "<<g<<endl;
for(ll i=;i*i<=g;i++){
if(g%i==){
fac[++tot]=i;
if(i!=g/i) fac[++tot]=g/i;
}
}sort(fac+,fac+tot+);
for(int i=;i<=tot;i++){
//cout<<fac[i]<<" ";
bool fl=true;
for(int j=;j<=k-;j++){
if(a[j]%fac[i]==){
fl=false;break;
}
}
if(fl){
printf("%lld",n/fac[i]);return ;
}
}
//cout<<" over "<<endl;
return ;
}
然而这个代码BZOJ过不去。
还要优化。
发现,每次对一个因数判断所有的k个数代价太大。
一个a[i]影响的是哪些gcd(a[k],n)的因数呢?
一定是gcd(a[i],a[k])的因数。
所以,我们可以把所有的gcd(a[i],a[k])的因数找出来干掉。
然后直接再扫一遍,找到最小的没有被干掉的因数即可。
复杂度?O(sqrt(n)+klogn+玄学)
代码:
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#include<bits/stdc++.h>
#define ri register int
#define numb (ch^'0')
using namespace std;
typedef long long ll;
const int N=+;
void rd(ll &x){
x=;
char ch;
while(!isdigit(ch=getchar()));
for(x=numb;isdigit(ch=getchar());x=(x<<)+(x<<)+numb);
}
ll n,k;
ll gcd(ll a,ll b){
return b?gcd(b,a%b):a;
}
ll a[N],fac[N];
bool kil[N];
int tot;
int main(){
scanf("%lld%lld",&n,&k);
for(ri i=;i<=k;i++)scanf("%lld",&a[i]);
ll g=gcd(a[k],n);
for(ri i=;i<k;++i) a[i]=gcd(a[i],g);
for(ll i=;i*i<=g;i++){
if(g%i==){
fac[++tot]=i;
if(i!=g/i) fac[++tot]=g/i;
}
}sort(fac+,fac+tot+);
for(ri i=;i<k;i++){
kil[lower_bound(fac+,fac+tot+,a[i])-fac]=;
}
for(ri i=;i<=tot;++i){
if(kil[i]){
for(int j=;j<i;j++) {
if(fac[i]%fac[j]==) kil[j]=;
}
}
}
for(ri i=;i<=tot;++i){
if(!kil[i]) {
printf("%lld\n",n/fac[i]);
return ;
}
}
return ;
}
[POI2011]SEJ-Strongbox的更多相关文章
- bzoj 2277 [Poi2011]Strongbox 数论
2277: [Poi2011]Strongbox Time Limit: 60 Sec Memory Limit: 32 MBSubmit: 527 Solved: 231[Submit][Sta ...
- bzoj2277 [Poi2011]Strongbox
2277: [Poi2011]Strongbox Time Limit: 60 Sec Memory Limit: 32 MBSubmit: 498 Solved: 218[Submit][Sta ...
- BZOJ2277[Poi2011]Strongbox——数论
题目描述 Byteasar is a famous safe-cracker, who renounced his criminal activity and got into testing and ...
- [poi2011]bzoj 2277 —— strongbox·[洛谷3518]
·问题描述· 有一个密码箱,0到n-1中的某些数是它的密码.且满足:如果a和b都是它的密码,那么(a+b)%n也是它的密码.某人试了k次密码,前k-1次都失败了,最后一次成功. 问:该密码箱最多有多少 ...
- 【BZOJ】2277: [Poi2011]Strongbox
题意 有一个密码箱,\(0\)到\(n-1\)中的某些整数是它的密码.如果\(a\)和\(b\)都是它的密码,那么\((a+b)%n\)也是它的密码(\(a,b\)可以相等).某人试了\(k\)次密码 ...
- BZOJ 2277 Poi2011 Strongbox
题目大意:一个集合A,包含了0~n-1这n个数.另有一个集合B,满足: 1.B是A的子集. 2.如果a.b均在B中,则(a+b)%n也在B中(a=b亦可) 给出k个数ai,前k-1个不在B中,第k个在 ...
- BZOJ2277 [Poi2011]Strongbox 【数论】
题目链接 BZOJ2277 题解 orz太难了 如果一个数\(x\)是密码,那么所有\((x,n)\)的倍数都是密码 如果两个数\(x,y\)是密码,那么所有\((x,y)\)的倍数都是密码 那么如果 ...
- POI2011题解
POI2011题解 2214先咕一会... [BZOJ2212][POI2011]Tree Rotations 线段树合并模板题. #include<cstdio> #include< ...
- BZOJ2527: [Poi2011]Meteors
补一发题解.. 整体二分这个东西,一开始感觉复杂度不是很靠谱的样子 问了po姐姐,说套主定理硬干.. #include<bits/stdc++.h> #define ll long lon ...
- BZOJ2276: [Poi2011]Temperature
2276: [Poi2011]Temperature Time Limit: 20 Sec Memory Limit: 32 MBSubmit: 293 Solved: 117[Submit][S ...
随机推荐
- 记录阿里云ECS(Centos7.4)安装mysql 8.0.X服务
#*.rpm介绍 大多数二进制rpm包都包含在名称中倒数第二个字段中编译rpm的体系结构..rpm软件包有那么几种 *.src.rpm 源程序包,要先通过编译才能安装 *.noarch.rpm 该包适 ...
- CentOS7.2最小化安装后系统优化
系统初始化技术的演变 1.sysvinit技术 (1)Linux系统的第一个进程(pid=1)为init: Linux 操作系统的启动首先从 BIOS 开始,接下来进入 boot loader,由 b ...
- jenkins升级为2.134
由于前面装的jenkins版本为2.130版本,昨天(2018.7.26)发现了两个jenkins的漏洞,影响范围为:Jenkins weekly 2.132 以及更早的版本.Jenkins LTS ...
- 云主机启动提示Booting from Hard Disk GRUB
版本:Openstack ocata 系统:centos7.3 环境:VMware workstation12 解决方法: 或者
- ASCII码中的可见字符
ASCII码中 包括空格的可见字符从32到126共95个 不包括则为94
- ES6的新特性(8)——数组的扩展
数组的扩展 扩展运算符 含义 扩展运算符(spread)是三个点(...).它好比 rest 参数的逆运算,将一个数组转为用逗号分隔的参数序列. console.log(...[1, 2, 3]) / ...
- Python 并行分布式框架:Celery 超详细介绍
本博客摘自:http://blog.csdn.net/liuxiaochen123/article/details/47981111 先来一张图,这是在网上最多的一张Celery的图了,确实描述的非常 ...
- GCD最大公约数
说明: 最初跟鹏哥学习最大公约数的算法是辗转相除,确实印象很深刻,那种辗转赋值的思想在好多题目中都有运用,但随着进一步学习,我也参考了其他几种方便快捷的最大公约数求法,在这里做一个总结. . int ...
- scrum立会报告+燃尽图(第二周第四次)
此作业要求参考: https://edu.cnblogs.com/campus/nenu/2018fall/homework/2249 一.小组介绍 组名:杨老师粉丝群 组长:乔静玉 组员:吴奕瑶.公 ...
- unknown2
结对作业 本次结对:211606457 郑沐榕.211406242 杨长元 一.预估与实际 PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时( ...