题目

https://www.luogu.com.cn/problem/P4571

思路

首先观察并且简单模拟一下火星人取燃料的过程,发现最终燃料的量一定是他选的k个瓶子容量的线性组合(观察操作3就知道)。火星人的抠门,就是说他会找到这些线性组合当中最小的正整数结果\(y\)。

让我们用式子描述一下,设k个瓶子的容量分别为\(x_1,x_2,...x_k\),那么有不定方程\(a_1x_1+a_2x_2+...+a_kx_k=y\),我们要求的就是使方程有解的最小的\(y\)。

然后根据裴蜀定理,\(y_{min}=gcd(x_1,x_2,...,x_k)\),题目就变成了要选k个数使得他们gcd值最大。

先不考虑时间复杂度,看看怎么保证正确性,首先gcd一定是某个\(a[i]\)的因数,那我们就对每个\(a[i]\)求一遍因数,然后该因数所对应的计数器++(我们先不考虑怎样计数)。最后查一下计数器\(\geq k\)的值。

考虑下总共有多少因数,一个数的除数函数\(\sigma_0(a)\)有个不紧的上界\(2\sqrt a\),总共1000个数,每个数因数个数小于1e5(事实上肯定远远小于),所以我们写个哈希再开个数组计数就行了。

然后就是时间复杂度的问题。求一个数的因数要实打实的\(\sqrt a\),总计是\(10^8\)级别,就很容易挂,得想办法优化一下。

①对于质数,它只有本身是有贡献的,根本不用继续试除,我们写个miller_rabin把素数判掉。

②维护一个全局ans值保存当前最大合法gcd值,如果在试除过程中不可能再有因子大于ans了就跳出。

③预先将a数组大到小排序,可能有助于提前找到很大的最优解。

加了这几个优化之后跑得飞快(287ms)。

代码

点击查看代码
#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<cmath>
#define ll long long
using namespace std;
int prime[10]={0,2,3,5,7,11};
int n,k,ans=0;
int a[1010];
const int zzd=233333;
struct Hashmap{
int fst[zzd],nxt[1000000],cnt,val[1000000];
Hashmap(){cnt=0;}
int insert(int x){
nxt[++cnt]=fst[x%zzd];
val[cnt]=x;
fst[x%zzd]=cnt;
return cnt;
}
int find(int y){
int i=fst[y%zzd];
while(i){
if(val[i]==y) return i;
i=nxt[i];
}
return 0;
}
} H;
int c[1000000];
ll qpow(ll x,ll p,ll m){
ll ans=1,base=x;
for(;p;p>>=1){
if(p&1) ans=ans*base%m;
base=base*base%m;
}
return ans;
}
int miller_rabin(ll x){
if(x==1) return 0;
ll y=x-1;
int i,j,u=0;
while(!(y&1ll)) y>>=1,u++;
for(i=1;i<=5;++i){
if(x==prime[i]) return 1;
if(!(x%prime[i])) return 0;
ll q=qpow(prime[i],y,x);
for(j=0;j<=u;++j){
if((q*q)%x==1&&q!=1&&q!=x-1) return 0;
if(j==u&&q!=1) return 0;
q=q*q%x;
}
}
return 1;
}
bool cmp(ll x,ll y){
return x>y;
}
void dec(int y){
int i,z;
int x=y,lim=(int)sqrt(x)+1;
for(i=1;i<=lim;++i){
if(x%i) continue;
if(x/i<=ans) break;
z=H.find(i);
if(!z) z=H.insert(i);
if(++c[z]>=k) ans=max(ans,i); z=H.find(x/i);
if(!z) z=H.insert(x/i);
if(++c[z]>=k) ans=max(ans,x/i);
if(i==1&&miller_rabin(x)) break;
}
return;
}
int main(){
int i,j;
scanf("%d%d",&n,&k);
for(i=1;i<=n;++i) scanf("%d",&a[i]);
sort(a+1,a+n+1,cmp);
for(i=1;i<=n;++i){
if(a[i]<=ans) break;
dec(a[i]);
}
printf("%d",ans);
// system("pause");
return 0;
}

洛谷P4571 [JSOI2009] 瓶子和燃料的更多相关文章

  1. 洛谷 P4571 BZOJ 2257 [JSOI2009]瓶子和燃料

    bzoj题目链接 上面hint那里是选择第2个瓶子和第3个瓶子 Time limit 10000 ms Memory limit 131072 kB OS Linux Source Jsoi2009 ...

  2. 【BZOJ2257】[JSOI2009]瓶子和燃料(数论)

    [BZOJ2257][JSOI2009]瓶子和燃料(数论) 题面 BZOJ 洛谷 题解 很明显就是从\(n\)个数里面选\(K\)个数让他们的\(gcd\)最大. 暴力找所有数的因数,拿个什么东西存一 ...

  3. BZOJ 2257: [Jsoi2009]瓶子和燃料 裴蜀定理

    2257: [Jsoi2009]瓶子和燃料 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/p ...

  4. bzoj2257 [Jsoi2009]瓶子和燃料 最大公约数

    [Jsoi2009]瓶子和燃料 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1449  Solved: 889[Submit][Status][Di ...

  5. BZOJ 2257: [Jsoi2009]瓶子和燃料【数论:裴蜀定理】

    2257: [Jsoi2009]瓶子和燃料 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1326  Solved: 815[Submit][Stat ...

  6. bzoj2257: [Jsoi2009]瓶子和燃料

    2257: [Jsoi2009]瓶子和燃料 Time Limit: 10 Sec  Memory Limit: 128 MB Description jyy就一直想着尽快回地球,可惜他飞船的燃料不够了 ...

  7. [BZOJ 2257][JSOI2009]瓶子和燃料 题解(GCD)

    [BZOJ 2257][JSOI2009]瓶子和燃料 Description jyy就一直想着尽快回地球,可惜他飞船的燃料不够了. 有一天他又去向火星人要燃料,这次火星人答应了,要jyy用飞船上的瓶子 ...

  8. [JSOI2009]瓶子和燃料

    Description jyy就一直想着尽快回地球,可惜他飞船的燃料不够了. 有一天他又去向火星人要燃料,这次火星人答应了,要jyy用飞船上的瓶子来换.jyy 的飞船上共有 N个瓶子(1<=N& ...

  9. [JSOI2009]瓶子和燃料 BZOJ2257 数学

    题目描述 jyy就一直想着尽快回地球,可惜他飞船的燃料不够了.有一天他又去向火星人要燃料,这次火星人答应了,要jyy用飞船上的瓶子来换.jyy的飞船上共有 N个瓶子(1<=N<=1000) ...

  10. 【数学 裴蜀定理】bzoj2257: [Jsoi2009]瓶子和燃料

    使gcd最大的trick Description jyy就一直想着尽快回地球,可惜他飞船的燃料不够了. 有一天他又去向火星人要燃料,这次火星人答应了,要jyy用飞船上的瓶子来换.jyy的飞船上共有 N ...

随机推荐

  1. Java面向对象之封装详解

    封装详解 封装 该露的露,该藏的藏 1.我们程序设计要追求"高内聚.低耦合".高内聚:类的内部数据操作细节自己完成,不允许外部干涉:低耦合:仅暴露少量的方法给外部使用. 封装(数据 ...

  2. myJRebel 已不可用

    周末在家里撸代码,突然 IDEA 提示 JRebel 需要激活. 原来一直使用的 myJRebel 的激活码,天真的以为是我的网络问题,尝试重新激活,结果不管用,就想去 myJrebel 的网站上去看 ...

  3. memoの关于Qt的一些用法记录

    Qt自动调整窗口尺寸 之前写过,方法就是: QTimer::singleShot(0, this, [this]{ this->adjustSize(); }); 重复记录一下. 如何把一个Mo ...

  4. 初学银河麒麟linux笔记 第五章 windows中开发的QT程序适配linux的修改——外部控件重新调用

    本人在WINDOWS系统中使用了"飞扬青云"的控件 https://gitee.com/feiyangqingyun/QUCSDK 由于系统移植,调用库应改为linux系统,首先下 ...

  5. OS-lab3

    OS-lab3 lab2之后,我们能够通过MMU访问内存了,不过操作系统最重要的是能够让进程运行. include env.h 定义了进程控制相关的变量,如进程数量NENV.进程状态ENV_FREE等 ...

  6. Linux 第五节(特殊权限,隐藏权限,SU,SUDO,FHS文件系统层次化标准)

    特殊权限 SUID  执行者临时获取命令的所有权限(对程序进行设置) SGID  目录内新文件所有组,继承原有目录所有组的名称 SBID  粘滞位,保护位 chmod +权限  文件 chmod   ...

  7. 三、核心实战-服务Service-Ingress

    Service 将一组Pods公开为网络服务的抽象方法. 暴露deployment只能在集群内访问是ClusterIP,可以集群外访问是NodePort,默认端口分配是30000-32767之间 ku ...

  8. 通过Python获取cpu、硬盘和主板等硬件序列号组成的唯一识别码

    import wmi c = wmi.WMI() def yingpan(): # # 硬盘序列号 cc = "" for physical_disk in c.Win32_Dis ...

  9. eFuse技术

    1. 基本概况及介绍   不同于大多数FPGA使用的SRAM阵列,eFuse一次只有一根熔丝能够被编程,这是该方法的配置能力存在限制范围的原因.但当与日益成熟的内置自测试(BIST)引擎组合使用时,这 ...

  10. 【36oj】 画圣诞树

    原题 圣诞节要到了,不少商家在宣传板上绘制了圣诞树的图案,如图所示.一棵圣诞树由A和B两部分组成: A是由n(n≥)个呈三角形的字符矩阵构成的,每个字符矩阵由三个参数ai.bi.ci唯一确定.Ai表示 ...