题目描述
韩信点兵,多多益善。
战前,萧何备军。韩信检阅,曰:“增1人?”。萧何答:“甚好”。韩信再曰:“使看齐?”。
萧何再答“甚好”。
输入
输入有多组(不超过100)测试实例。
每组测试实例第1行为1个正整数M(1 ≤ M ≤ 100),表示接下来有M行士兵的信息。每行士兵信息为两个正整数H(100 <= H <= 200),N(1 <= N <= 1000),分别表示士兵身高和其对应的士兵人数。
输入结束将由一行M=0的测试实例表示,不应处理此测试实例。
输出
每组测试实例输出一行,格式为:“Case i: sum”,其中i是测试实例的编号(从1开始),sum为一个正整数,即增加的这名士兵与其他每名士兵的身高差之和,该和满足:是所有可能的和当中最小的和。
样例输入
2
160 1
170 1
4
160 1
165 1
170 1
180 2
0
样例输出
Case 1: 10
Case 2: 35
解析:题目其实就是求插入一个士兵,使得他与每个士兵的身高和达到最小时的,身高差总和。
其实,n个数中,求一个数到所有数差和最小,这个数就是这一列数的中位数,然后再求差和即可。
当然我们其实也可以直接暴力遍历,这题不会超时,但我们得知道,这个插入的士兵的身高数值肯定在原有士兵最矮和最高之间。我们可以利用一个数组来存储原有所有士兵的身高数据,然后遍历求最小值。在这里建议数组开大一点。
中位数:
#include <bits/stdc++.h>
using namespace std;
int k[100005];
int main()
{int m,n,i,s,mid,cnt=0,sg,ge,sum;while(~scanf("%d",&n)){ //n种身高的士兵 if(n==0) break;m=0,sum=0,cnt++; //cnt用来计数case for(i=0;i<n;i++){scanf("%d %d",&sg,&ge); //输入身高和对应个数 for(s=0;s<ge;s++) k[m++]=sg; //存入数组 }sort(k,k+m); //排序 if(m%2==1) mid=k[m/2]; //求出中位数mid else mid=k[m/2-1];for(i=0;i<m;i++){sum=sum+abs(k[i]-mid); //累加 }printf("Case %d: %d\n",cnt,sum);}return 0;
}
暴力:
#include <stdio.h>
int k[1000005];
int main()
{long long n,i,s,m=0,a,b,max=100,min=200,sum=0,cnt=1,min1=9999999;while(~scanf("%lld",&n)){m=0;if(n==0){break;}for(i=0;i<n;i++){scanf("%lld %lld",&a,&b);if(a>max) max=a;if(a<min) min=a;for(s=0;s<b;s++){k[m]=a; //存入k数组中 m++;}}for(i=min;i<=max;i++){for(s=0;s<m;s++){sum=sum+abs(i-k[s]); //求身高差和 }if(sum<min1) min1=sum; //如比最小值小,替换 sum=0;}printf("Case %lld: %lld\n",cnt,min1);cnt++,min1=9999999; //初始化数据,cnt不用,记录第几个测试用例 max=100,min=200;}return 0;
}