ZOJ 3987 Numbers CCPC2017 Qinhuangdao(贪心+Java大整数)

 


        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);
		}
	}
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值