问题描述
试题编号: | 202303-2 | |||||||||||||||||||||||||
试题名称: | 垦田计划 | |||||||||||||||||||||||||
时间限制: | 1.0s | |||||||||||||||||||||||||
内存限制: | 512.0MB | |||||||||||||||||||||||||
问题描述: | 问题描述顿顿总共选中了 n 块区域准备开垦田地,由于各块区域大小不一,开垦所需时间也不尽相同。据估算,其中第 i 块(1≤i≤n)区域的开垦耗时为 ti 天。这 n 块区域可以同时开垦,所以总耗时 tTotal 取决于耗时最长的区域,即:tTotal=max{t1,t2,⋯,tn} 为了加快开垦进度,顿顿准备在部分区域投入额外资源来缩短开垦时间。具体来说:
现在顿顿手中共有 m 单位资源可供使用,试计算开垦 n 块区域最少需要多少天? 输入格式从标准输入读入数据。 输入共 n+1 行。 输入的第一行包含空格分隔的三个正整数 n、m 和 k,分别表示待开垦的区域总数、顿顿手上的资源数量和每块区域的最少开垦天数。 接下来 n 行,每行包含空格分隔的两个正整数 ti 和 ci,分别表示第 i 块区域开垦耗时和将耗时缩短 1 天所需资源数量。 输出格式输出到标准输出。 输出一个整数,表示开垦 n 块区域的最少耗时。 样例输入1
Data 样例输出1 样例解释如下表所示,投入 5 单位资源即可将总耗时缩短至 5 天。此时顿顿手中还剩余 4 单位资源,但无论如何安排,也无法使总耗时进一步缩短。
样例输入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;
}
}