标签:style blog http color os io for art
hdu4430:http://acm.hdu.edu.cn/showproblem.php?pid=4430
题意:题目的意思就是给你一个s,让你求k,r,其中k,r,满足:k^1+k^2+.....+k^r==s||s-1,并且k*r是最小的,如果有多种情况,r是最小的。
题解:一开始不知道怎么做,听别人说是二分。然后开始往二分上想。
思维过程:一开始就想着怎么二分,首先二分必须找到一个线性关系,找了半天也没有找到什么线性关系,所以不知道怎么二分,也不知道二分什么变量。想了很久,最后看到k*r最小,k是底数,r是指数,所以当r最大时候,k是最小的,同时k*r是最小的,因为k是指数增长的,,当r很大,k只要很小就可以达到s.当k==2的时候,r最多只要40就可以达到s。所以可以枚举r了,然后来找k。这样就慢慢想到,对于固定r,k越大,它的次方和也是随着增大的,所以线性关系出来了,可以二分k,k初始值是s||s-1,所以分两种情况来做,就可以了。不过,这里还要处理一些益处问题,在计算次方和的时候,如果底数大于1e7,其他的值也行,只要不小于1e6就直接跳出,不用计算。具体的,代码中有注释。
1 #include<iostream> 2 #include<cstring> 3 #include<cstdio> 4 #include<algorithm> 5 using namespace std; 6 long long n; 7 long long k,ans1,lastr,lastk,l,mid,r; 8 bool judge(long long s,int num,long long flag){ 9 long long ans=0,temp=s; 10 if(s>10000000)return true;//这里是特判,方式爆出long long 因为如果这个数>1e7的话,她的平方必然大于n(考虑n范围) 11 for(int i=1;i<=num;i++){ 12 ans+=temp; 13 temp*=s; 14 if(ans>=flag)return true;//这里也是防止溢出 15 } 16 return false; 17 } 18 bool judge2(long long s,int num){//判断这个数是不是要找的数 19 long long ans=0,temp=s; 20 if(s>10000000)return false; 21 for(int i=1;i<=num;i++){ 22 ans+=temp; 23 temp*=s; 24 if(ans>n)return false; 25 } 26 if(ans==n||ans==n-1)return true; 27 return false; 28 } 29 int main(){ 30 31 while(~scanf("%I64d",&n)){ 32 ans1=n-1;lastr=1;lastk=n-1; 33 for(int i=2;i<=40;i++){ 34 l=1,r=n; 35 while(l<r){//二分的结果是要么找到了==n的数,要么循环退出 36 mid=(l+r)/2; 37 if(judge(mid,i,n)) 38 r=mid; 39 else 40 l=mid+1; 41 } 42 if(judge2(l,i)&&ans1>i*l){//检查是否是找到==n的那个数 43 ans1=i*l; 44 lastr=i; 45 lastk=l; 46 } 47 l=1,r=n-1; 48 while(l<r){//要么找到==n-1的数,要么退出循环 49 mid=(l+r)/2; 50 if(judge(mid,i,n-1)) 51 r=mid; 52 else 53 l=mid+1; 54 } 55 if(judge2(l,i)&&ans1>i*l){//同上 56 ans1=i*l; 57 lastr=i; 58 lastk=l; 59 } 60 } 61 printf("%I64d %I64d\n",lastr,lastk); 62 } 63 }
Yukari's Birthday,布布扣,bubuko.com
标签:style blog http color os io for art
原文地址:http://www.cnblogs.com/chujian123/p/3877428.html