ZCMU--5193: 韩信点兵(C语言)

题目描述

韩信点兵,多多益善。

战前,萧何备军。韩信检阅,曰:“增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;
}

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值