题目链接:https://vjudge.net/problem/UVALive-6434

题意:给你n个数字,要你把这n个数字分成m组,每一组的消耗值定义为改组最大值和最小值之差,要求这m组的消耗值总和最小。

思路:既然我们要求把它分成m组,并且各组的最大值和最小值之差的和最小,我们就可以先把它分成n组,这时的总消耗值为零,我们为了让它的值尽可能的小,我们就可以每次选择两组使得总消耗值增加最少的

两组合并在一起,这时就只有n-1组了,重复以上操作n-m次,就可以得到分成m组所需的最小消耗了。(每次找寻使得总消耗增加最少的两组时,我们只要找到相邻的两组最大值和最小值差最小的就可以了)

假设当前 分组为(3,4)(7,8)

当我们要把这两组和并的时候,我们只要把总消耗值加上第二组的最小值和第一组的最大值的差就可以了,为什么可以这样呢?(当求一组数的最大值和最小值之差的和,不就可以看作是要把这组中的所有数字按照顺序连接在一起,每次连接的消耗

等于每两个相邻数字的差值,总和就是把他们全加其起来,例:(1,6,4,5,7,13,9),他们的最大值和最小值之差为12,不就等于先排序后(1,4,5,6,7,9,13),(4-1)+(5-4)+(6-5)+(7-6)+(9-7)+(13-9)=12  ,对于每组的连接也是相似的,只要把组看作单个数字就可以了

,只是它对于它前面一组的值是本组的最小值,对于后面一组的值为本组的最大值  )

我们可以看作把3,4和成一块消耗值为1,把7,8和成一块消耗值为1,这时3,4可以看作是一块,7,8可以看作是一块,当你要把3,4和7,8这两块连接在一起的时候,要增加的消耗不就是把4和7连接在一起的消耗吗。

#include<cstdio>
#include<algorithm>
#include<cstring>
#define INF 1e9+5
using namespace std;
int s[110];
int vt[110];//记录当前位置的数字是属于哪个集合的
int main(){
int t;
int n,m;
int i,j,k;
int a,b,c;
int mn;
int nb=0;
scanf("%d",&t);
while(t--){
nb++;
scanf("%d%d",&n,&m);
for(i=0;i<n;i++){
scanf("%d",&s[i]);
vt[i]=i;//分成n组
}
if(m==n)
printf("Case #%d: 0\n",nb);
else{
sort(s,s+n);
for(i=n;i>m;i--){
mn=INF;
a=vt[0];
for(j=1;j<n;j++){
if(vt[j]!=a){//两个分组的交界位置
if(s[j]-s[j-1]<=mn){//找把两个集合合并在一起增加消耗最小的位置 ,一开始的思路只是找出合并后最大值和最小值之差最小的两个组进行合并
//结果wa了一发,仔细想想后发现这样合并后并不能把增加的消耗值降到最小
/*
1
10 2
1 5 6 7 8 9 10 13 13 13
当分组数位3的时候
此时分组为 (1) (5,6,7,8,9,10) (13,13,13)
如果只是单纯的把合并后最大值和最小值之差最小的两组合并的话,
(1,5,6,7,8,9,10) (13,13,13)
此时总消耗为9
(1) (5,6,7,8,9,10,13,13,13)
而最小的总消耗却是8
不能保证每次的增长量最小
*/
c=j;//记录分界点
mn=s[j]-s[j-1];
}
a=vt[j];//连接相邻的两个组的消耗一定是最小的
}
}
/*for(j=0;j<n;j++){//不太理解的同学可以输出一下每次的分组情况便于理解
printf("%d ",vt[j]);
}
printf("\n");*/
int y=vt[c];
for(j=c;vt[j]==y;j++){//合并
vt[j]=vt[c-1];
}
}
a=vt[0];
b=0;
long long sum=0;
for(i=1;i<n;i++){
if(vt[i]!=a){
sum+=s[i-1]-s[b];//计算消耗值
b=i;
a=vt[i];
}
}
sum+=s[n-1]-s[b];
printf("Case #%d: %lld\n",nb,sum);
} }
return 0;
}

  

#include<stdio>
#include<algorithm>
using namespace std;
int s[110]
int main(){
int t,n,m;
int i,j,nb;
nb=0;
int a,b;
scanf("%d",&t);
while(t--){
nb++;
scanf("%d%d",&n,&m);
for(i=0;i<n;i++){
scanf("%d",&s[i]);
}
sort(s,s+n);
for(i=0;i<n-1;i++)//直接记录下所有相邻的数字要合并在一起的消耗(这和上个代码的核心思想一致,但是
//少了许多不必要的操作,直接记录下最核心的数据)
s[i]=s[i+1]-s[i];
sort(s,s+n-1);
int sum=0;
for(i=0;i<n-m;i++){//每次找到消耗代价最小的那一次合并
sum+=s[i];
}
printf("Case #%d: %d",nb,sum);
}
return 0;
}

  

UVALive - 6434 (贪心)的更多相关文章

  1. UVALive - 3266 (贪心) 田忌赛马

    耳熟能详的故事,田忌赛马,第一行给出田忌的马的速度,第二行是齐王的马的速度,田忌赢一场得200,输一场失去200,平局不得也不失,问最后田忌最多能得多少钱? 都知道在故事里,田忌用下等马对上等马,中等 ...

  2. UVALive - 6434 (思维题)

    题目链接:https://vjudge.net/contest/241341#problem/A 题目大意,给你n个数字,让你分成m组,每组的花费为每组的最大值-最小值,总的花费就是各组花费相加,要求 ...

  3. UVALive - 6434 —(思维题)

    题意:给出了你由n个数组成的序列,让你将这个序列分为成m个集合,使得每一个集合的最大值减最小值的差相加最小.(如果某集合只有一个数字,则最大值减最小值为0) . 思路:首先我们不难想到,最优的分配方法 ...

  4. UESTC 2016 Summer Training #6 Div.2

    我好菜啊.. UVALive 6434 给出 n 个数,分成m组,每组的价值为最大值减去最小值,每组至少有1个,如果这一组只有一个数的话,价值为0 问 最小的价值是多少 dp[i][j] 表示将 前 ...

  5. Gym 101194D / UVALive 7900 - Ice Cream Tower - [二分+贪心][2016 EC-Final Problem D]

    题目链接: http://codeforces.com/gym/101194/attachments https://icpcarchive.ecs.baylor.edu/index.php?opti ...

  6. 贪心 UVALive 6834 Shopping

    题目传送门 /* 题意:有n个商店排成一条直线,有一些商店有先后顺序,问从0出发走到n+1最少的步数 贪心:对于区间被覆盖的点只进行一次计算,还有那些要往回走的区间步数*2,再加上原来最少要走n+1步 ...

  7. 贪心 UVALive 6832 Bit String Reordering

    题目传送门 /* 贪心:按照0或1开头,若不符合,选择后面最近的进行交换.然后选取最少的交换次数 */ #include <cstdio> #include <algorithm&g ...

  8. UVALive 7147 World Cup(数学+贪心)(2014 Asia Shanghai Regional Contest)

    题目链接:https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&category=6 ...

  9. UVALive 7146 Defeat the Enemy(贪心+STL)(2014 Asia Shanghai Regional Contest)

    Long long ago there is a strong tribe living on the earth. They always have wars and eonquer others. ...

随机推荐

  1. Axios的默认配置(碎片知识)API

    axios API axios(config) axios({ method: 'Post', url: '/user/123', data: { //略 } }) axios(url[, confi ...

  2. mandatory and advisory文件锁(File Lock)

    http://blog.csdn.net/elfprincexu/article/details/43564425 文件锁(File Lock)是一种在特定的时间内只允许一个进程进行访问文件的机制,通 ...

  3. Confluence 6 指派和撤销空间权限

    指派空间权限 希望添加一个新用户或者用户组到权限列表中,从希望选择的选项中查找用户组或者用户,然后选择 添加(Add).用户和用户组将会显示在列表中:选择你希望引用的权限,然后选择 保存所有(Save ...

  4. Vue之单文件组件的数据传递,axios请求数据及路由router

    1.传递数据 例如,我们希望把父组件的数据传递给子组件. 可以通过props属性来进行传递. 传递数据三个步骤: 步骤1:在父组件中,调用子组件的组名处,使用属性值的方式往下传递数据 <Menu ...

  5. python基础之函数参数,名称空间,以及函数嵌套

    函数进阶内容梗概: 1. 函数参数--动态传参 2. 名称空间, 局部名称空间, 全局名称空间, 作⽤用域, 加载顺序. 3. 函数的嵌套 4. gloabal , nonlocal 关键字 1. 函 ...

  6. Fiddler 4 界面功能介绍

    由于本人工作接触Web测试,所以我从网上找的资料,学习了解web测试内容,然后自己整理汇总的随笔,如文章中有不足的地方,请大家多多指教:或者文章内容与他人相似,望见谅 主界面: 工具栏 file:用于 ...

  7. P4238 【模板】多项式求逆 ntt

    题意:求多项式的逆 题解:多项式最高次项叫度deg,假设我们对于多项式\(A(x)*B(x)\equiv 1\),已知A,求B 假设度为n-1,\(A(x)*B(x)\equiv 1(mod x^{\ ...

  8. 把javabean复制到另一个javabean 使用BeanUtils.copyProperties(a,b) 复制

    该方法对于两种不同的jar包有两种不同的意义 ,a,b通常是两个结构相似的javabean,注意:a,b里的定义类型名称必须一致才能复制 引用的是org.springframework.beans 则 ...

  9. Linux NTP服务配置 for Oracle RAC

    安装Oracle 11g RAC时,我们需要配置ntp服务.在使用虚拟机的情况下对于时钟同步方式的配置有很多种方式,可以使用vmware自带的时钟同步功能,也可以直接将本地的一个节点用作时间服务器.本 ...

  10. 主流Linux发行版简介

    发行版 包格式 所属厂商 发布年份 最新版本 说明 RHEL RPM RedHat 2004 7.3 RedHat Linux9.0后的企业方向,系统可以免费下载安装但无法使用官方yum源 Fedor ...