题目

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)。

代码

点击查看代码
  1. #include<cstdio>
  2. #include<cstdlib>
  3. #include<algorithm>
  4. #include<cmath>
  5. #define ll long long
  6. using namespace std;
  7. int prime[10]={0,2,3,5,7,11};
  8. int n,k,ans=0;
  9. int a[1010];
  10. const int zzd=233333;
  11. struct Hashmap{
  12. int fst[zzd],nxt[1000000],cnt,val[1000000];
  13. Hashmap(){cnt=0;}
  14. int insert(int x){
  15. nxt[++cnt]=fst[x%zzd];
  16. val[cnt]=x;
  17. fst[x%zzd]=cnt;
  18. return cnt;
  19. }
  20. int find(int y){
  21. int i=fst[y%zzd];
  22. while(i){
  23. if(val[i]==y) return i;
  24. i=nxt[i];
  25. }
  26. return 0;
  27. }
  28. } H;
  29. int c[1000000];
  30. ll qpow(ll x,ll p,ll m){
  31. ll ans=1,base=x;
  32. for(;p;p>>=1){
  33. if(p&1) ans=ans*base%m;
  34. base=base*base%m;
  35. }
  36. return ans;
  37. }
  38. int miller_rabin(ll x){
  39. if(x==1) return 0;
  40. ll y=x-1;
  41. int i,j,u=0;
  42. while(!(y&1ll)) y>>=1,u++;
  43. for(i=1;i<=5;++i){
  44. if(x==prime[i]) return 1;
  45. if(!(x%prime[i])) return 0;
  46. ll q=qpow(prime[i],y,x);
  47. for(j=0;j<=u;++j){
  48. if((q*q)%x==1&&q!=1&&q!=x-1) return 0;
  49. if(j==u&&q!=1) return 0;
  50. q=q*q%x;
  51. }
  52. }
  53. return 1;
  54. }
  55. bool cmp(ll x,ll y){
  56. return x>y;
  57. }
  58. void dec(int y){
  59. int i,z;
  60. int x=y,lim=(int)sqrt(x)+1;
  61. for(i=1;i<=lim;++i){
  62. if(x%i) continue;
  63. if(x/i<=ans) break;
  64. z=H.find(i);
  65. if(!z) z=H.insert(i);
  66. if(++c[z]>=k) ans=max(ans,i);
  67. z=H.find(x/i);
  68. if(!z) z=H.insert(x/i);
  69. if(++c[z]>=k) ans=max(ans,x/i);
  70. if(i==1&&miller_rabin(x)) break;
  71. }
  72. return;
  73. }
  74. int main(){
  75. int i,j;
  76. scanf("%d%d",&n,&k);
  77. for(i=1;i<=n;++i) scanf("%d",&a[i]);
  78. sort(a+1,a+n+1,cmp);
  79. for(i=1;i<=n;++i){
  80. if(a[i]<=ans) break;
  81. dec(a[i]);
  82. }
  83. printf("%d",ans);
  84. // system("pause");
  85. return 0;
  86. }

洛谷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. 【面试题】ES6语法五之箭头函数

    ES6特性=>. function foo(x, y){ return x + y } var foo = (x, y) => x + y 包括一个参数列表(零个或多个参数,如果参数不是一 ...

  2. MobaXterm激活专业版

    本文思路来自 https://github.com/flygon2018/MobaXterm-keygen 有python 环境 并且不看英文的可以继续往下 不然直接访问这个地址也行. 1.需要一个大 ...

  3. jar包下不下来

    1.maven中的settings.xml文件中的镜像资源配置 <mirror> <id>alimaven</id> <name>aliyun mave ...

  4. Chrome禁用开发者工具

    在一次工作中,所做的项目要求页面中不能右击,不能打开F12.一般来说可以禁用F12的按键,但是可以通过开发者工具进入.经过个人实验,以下方法适用于谷歌浏览器.火狐浏览器,以及使用谷歌内核的浏览器(如Q ...

  5. .net core layui折叠表格的应用。

    效果展示 头部的折叠,展开,搜索按钮 <div class="layui-fluid"> <div style="margin-top: 20px;&q ...

  6. 合格できる日本語能力試験, N1.PDF

    书本详情 合格できる日本語能力試験, N1种类:Languages - Japanese Language Reference年:2010出版:Shohan.出版社:Aruku语言:japanese页 ...

  7. 基于excel的自动化框架

    设定项目文件大致结构 atp/: 项目名 conf/:存放配置文件 data/:存放sql文件 lib/: 存放项目的所有源代码. logs/:存放日志文件 uploads/:存放下载的文件 star ...

  8. display和浮动

    display:block; // 块元素 display: inline-block //行内和块元素 浮动 float: left; 清除浮动 clear: both; 两侧不允许有浮动 解决父级 ...

  9. nodeJs 写个爬虫小玩意

    内容 起一个服务,爬某个网站的数据(我这里爬了个夕阳红游戏交易网站的数据),页面看到我要爬的内容 代码 1 //引入内置的http包 2 var http = require('http'); 3 c ...

  10. 【当年笔记】Collection集合部分

    集合继承关系图 1)Vector 特点:线程安全,消耗偏大 2)ArrayList 特点:基于数组实现,随机访问某个元素效率高.集和头尾之间包括头插入删除操作效率较低,因为插入元素后,其他元素要后移. ...