题目地址:
http://acm.hdu.edu.cn/showproblem.php?pid=4430
题目:
18 111 1111
1 17 2 10 3 10
题意:
将n支生日蛋糕的蜡烛像同心圆那样围成一圈一圈的,最里面的那个圈只能是1支蜡烛,其他圈按k^i(i=1~r)的数量增长,问给定n支,求实r*k最小的r,k序列,如果有多个r*k值相同的最小值,进一步求r最小的序列。
题解:
抽象成数学问题就是:有符合以下条件的式子:k^0 + k^1 + k^2 + k^3 +......+k^r=n,求 对任意的r,k使r*k最小。注意下取值范围,k是>=2的 n<=10^12,那么很容易能估算出r最大值在40左右。时间要求也很充足,可以一次穷举r和k 以此求出最小值,这里r顺序枚举,在确定k值时用二分枚举,直接用公式求k不是很好求,并且要考虑浮点数误差的问题,这点很容易导致无限的WA中。
代码:
/* hdu :Yukari's Birthday */ #include<stdio.h> #include<stdlib.h> #include<string.h> #include<math.h> #include<iostream> #include<string> using namespace std; __int64 R=0,K=0,Value=0;//R 's max is about 40 __int64 MinR=0,MinK=0,MinValue=0; __int64 n=0; /*init the var*/ int InitVar() { MinR=1; MinK=n-1; MinValue=n-1; return(0); } /*for test*/ int test() { return(0); } /*main process*/ int MainProc() { while(scanf("%I64d",&n)!=EOF) { InitVar(); for(R=2;R<=40;R++) { __int64 KLow=2; __int64 KHigh=(__int64)pow((double)n,1.0/(double)R); while(KLow<=KHigh) { __int64 KMid=(KLow+KHigh)/2; //__int64 N=(__int64)((pow((double)KMid,(double)(R+1))-1)/(double)(KMid-1)+0.0001); __int64 N=0; int i=0; __int64 RR=1; for(i=0;i<R;i++) { RR*=KMid; N+=RR; } if(N==n||N==n-1) { K=KMid; Value=K*R; break; } else if(N>n)//K is too big { KHigh=KMid-1; } else { KLow=KMid+1; } } if(Value<=MinValue&&KLow<=KHigh) { if(Value<MinValue) { MinR=R; MinK=K; MinValue=Value; } else { if(R<MinR) { MinR=R; MinK=K; MinValue=Value; } } } } printf("%I64d %I64d\n",MinR,MinK); } return(0); } int main() { MainProc(); return(0); }