csp试题 202303-2 垦田计划(Java)

问题描述

试题编号:202303-2
试题名称:垦田计划
时间限制:1.0s
内存限制:512.0MB
问题描述:

问题描述

顿顿总共选中了 n 块区域准备开垦田地,由于各块区域大小不一,开垦所需时间也不尽相同。据估算,其中第 i 块(1≤i≤n)区域的开垦耗时为 ti 天。这 n 块区域可以同时开垦,所以总耗时 tTotal 取决于耗时最长的区域,即:tTotal=max{t1,t2,⋯,tn}

为了加快开垦进度,顿顿准备在部分区域投入额外资源来缩短开垦时间。具体来说:

  • 在第 i 块区域投入 ci 单位资源,便可将其开垦耗时缩短 1 天;

  • 耗时缩短天数以整数记,即第 i 块区域投入资源数量必须是 ci 的整数倍;

  • 在第 i 块区域最多可投入 ci×(ti−k) 单位资源,将其开垦耗时缩短为 k 天;

  • 这里的 k 表示开垦一块区域的最少天数,满足 0<k≤min{t1,t2,⋯,tn};换言之,如果无限制地投入资源,所有区域都可以用 k 天完成开垦。

现在顿顿手中共有 m 单位资源可供使用,试计算开垦 n 块区域最少需要多少天?

输入格式

从标准输入读入数据。

输入共 n+1 行。

输入的第一行包含空格分隔的三个正整数 n、m 和 k,分别表示待开垦的区域总数、顿顿手上的资源数量和每块区域的最少开垦天数。

接下来 n 行,每行包含空格分隔的两个正整数 ti 和 ci,分别表示第 i 块区域开垦耗时和将耗时缩短 1 天所需资源数量。

输出格式

输出到标准输出。

输出一个整数,表示开垦 n 块区域的最少耗时。

样例输入1

4 9 2

6 1

5 1
6 2
7 1

Data

样例输出1

5

样例解释

如下表所示,投入 5 单位资源即可将总耗时缩短至 5 天。此时顿顿手中还剩余 4 单位资源,但无论如何安排,也无法使总耗时进一步缩短。

i基础耗时 ti缩减 1 天所需资源 ci投入资源数量实际耗时
16115
25105
36225
47125

样例输入2

4 30 2
6 1
5 1
6 2
7 1

样例输出2

2

样例解释

投入 20 单位资源,恰好可将所有区域开垦耗时均缩短为 k=2 天;受限于 k,剩余的 10 单位资源无法使耗时进一步缩短。

子任务

70% 的测试数据满足:0<n,ti,ci≤100 且 0<m≤106;

全部的测试数据满足:0<n,ti,ci≤105 且 0<m≤109。

 1.普通方法(70分):

import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        int n = in.nextInt();
        int m = in.nextInt();
        int k = in.nextInt();
        int[] t = new int[n];
        int[] c = new int[n];
        for( int i = 0; i < n; i++ ) {
        	t[i] = in.nextInt();
        	c[i] = in.nextInt();
        }

        int maxIndex = getMax(t);
        int max = t[ maxIndex ];
        while( m > 0 && max > k ) {
        	m -= c[maxIndex];
        	t[maxIndex] -= 1;
        	maxIndex = getMax(t);
        	max = t[maxIndex];
        }
        System.out.println(max);
    }
    
    //用来获取最大值的下标
    public static int getMax( int[] t ) {
    	int max = t[0];
    	int index = 0;
    	for( int i = 0; i < t.length; i++ ) {
    		if( t[i] > max ) {
    			max = t[i];
    			index = i;
    		}
    	}
    	return index;
    }
}

2.基于普通代码改进的二分法代码

import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        int n = in.nextInt();
        int m = in.nextInt();
        int k = in.nextInt();
        int[] t = new int[n];
        int[] c = new int[n];
        for( int i = 0; i < n; i++ ) {
        	t[i] = in.nextInt();
        	c[i] = in.nextInt();
        }
        int l = k, r = t[getMax(t)];
        while( l <= r ) {
        	int mid = (l + r)/2;
            //如果m可以满足当前的mid值,则说明最少天数在mid的左侧
        	if( check(mid, n, k, m, t, c) ) {
				r = mid - 1;
        	}else {
        		l = mid + 1;
			}
        	
        }
        System.out.println(l);
    }
    
    //用来判断当前mid的值是否能够实现
    //n,k,m,t,c可以设置为全局变量
    public static boolean check( int mid, int n, int k, int m, int[] t, int[] c) {
    	int sum = 0;
        //通过for循环寻找数组t中大于中间值mid的天数
    	for( int i = 0; i < n; i++ ) {
            //如果当前天数大于mid,就将其加到sum中
    		if( t[i] > mid  ) {
    			sum += ( t[i] - mid ) * c[i]; //sum为当前使用的资源数和,
    		}
    	}
        //如果当前sum值小于m,则说明当前情况可以实现,可以去将当前最小天数进一步压缩,返回true
    	if( sum <= m ) {
    		return true;
    	}
    	return false;
    }
    
    public static int getMax( int[] t ) {
    	int max = t[0];
    	int index = 0;
    	for( int i = 0; i < t.length; i++ ) {
    		if( t[i] > max ) {
    			max = t[i];
    			index = i;
    		}
    	}
    	return index;
    }
}

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值