CCPC秦皇岛的G题,当时几乎想出来怎么做了,但是需要用到一个大整数模板,要支持除法……然后暴力敲了几百行的c++程序,最后GG……
在这之后才知道,又一次的吃了Java的大亏,如果此题能够在1个小时内用Java大整数求出,那么后面的A题也能有更多的时间思考,做出了也没什么问题,然后就可能金牌了……
还是不说了,过了一个星期之后,在经历完青岛极其恶心的题目之后,有了三个星期的休息时间,于是不打算再次吃Java的亏,于是就简单的学了一下Java的基本操作和大整数类的东西。其实呢,和C++也差不多。
本题大致题意就是,给你一个大整数n,然后让你把这个整数分成m个数字,然后使得这m个数字的和为n而且这些数字的or最小。很明显的一道贪心的题目,我们直接考虑最后的结果。由于要求最后结果最小,我们直接看最后的结果在二进制位下,最高位是否可以取1。如果说后面的位都为1,这个数字乘以m比当前n剩余的要大,那么说明,当前二进制位可以不取1,根据要求,我们把这一位放作0。如果不满足上述条件,那么当前位置只能放1,放1的同时,我们要尽量利用这个1,于是就让所有的m个数字的这一个二进制位都放为1,这样n减少的就最多,后面就可以有更多的二进制位可以不放1。以此类推,一直重复即可。可以看出就是一个简单的贪心,难点就是在这个大整数的处理,而对于Java来说不存在这个问题。具体见代码:
import java.util.*;
import java.io.*;
import java.math.*;
public class Main {
public static BigInteger two=BigInteger.valueOf(2);
public static BigInteger p[]=new BigInteger[5000];
public static void init() //预处理二进制下每一位都为1的值
{
p[1]=BigInteger.ONE;
p[0]=BigInteger.ZERO;
for(int i=2;i<5000;i++)
p[i]=p[i-1].multiply(two);
for(int i=2;i<5000;i++)
p[i]=p[i].add(p[i-1]);
}
public static void main(String[] args)
{
init();
Scanner cin=new Scanner(System.in);
int T_T=cin.nextInt();
for(int T=1;T<=T_T;T++)
{
BigInteger ans=BigInteger.ZERO;
BigInteger a=cin.nextBigInteger();
BigInteger b=cin.nextBigInteger();
int up=0;
for(int i=0;i<5000;i++) //首先找到最高位
if (p[i].compareTo(a)>0) {up=i;break;}
for(int i=up;i>=1;i--)
{
ans=ans.multiply(two);
if (a.compareTo(p[i-1].multiply(b))<=0) continue; //若后面可以大于n剩余的量,那么这一位放0
BigInteger now=p[i].subtract(p[i-1]); //否则就只能放1,而且要让n尽量减去更多,剩下更少
BigInteger k=a.divide(now); ans=ans.add(BigInteger.ONE);
if (k.compareTo(b)>0) a=a.subtract(now.multiply(b));
else a=a.subtract(now.multiply(k));
}
System.out.println(ans);
}
}
}