UVA11300 Spreading the Wealth 题解
题目
A Communist regime is trying to redistribute wealth in a village. They have have decided to sit everyone around a circular table. First, everyone has converted all of their properties to coins of equal value, such that the total number of coins is divisible by the number of people in the village. Finally, each person gives a number of coins to the person on his right and a number coins to the person on his left, such that in the end, everyone has the same number of coins. Given the number of coins of each person, compute the minimum number of coins that must be transferred using this method so that everyone has the same number of coins.
一个村庄重新分配财富。 他们已经决定让每个人围坐在一张圆桌旁。 首先,每个人都将其所有财产转换为等值的硬币,这样硬币的总数就可被村庄中的人数整除。 最后,每个人都给右边的人一些硬币,给左边的人一些硬币,每个人都有相同数量的硬币。 给定每个人的硬币数量,计算必须使用此方法转移的最小硬币数量,以便每个人都拥有相同数量的硬币。
输入格式
There is a number of inputs. Each input begins with \(n (n < 1000001)\), the number of people in the village. n lines follow, giving the number of coins of each person in the village, in counterclockwise order around the table. The total number of coins will fit inside an unsigned 64 bit integer.
有多组输入。 每个输入均以数字\(n(n <1000001)\)开头,即人数.接下来\(n\)行,以逆时针方向给出村庄中每个人的硬币数量。 硬币总数将是一个无符号的64位整数。
输出格式
For each input, output the minimum number of coins that must be transferred on a single line.
对于每组输入,输出转移的最小硬币数。
输入样例
3
100
100
100
4
1
2
5
4
输出样例
0
4
题解
这道题感觉有点像数学题啊
因为一个人只能左右传硬币,所以只需要考虑左右
而每个人最开始的硬币数和最终的硬币数都是已知的,所以列方程即可
设\(coin_i\)为最开始每个人的硬币数量,\(pass_i\)为第\(i\)个人给逆时针方向的人(\(i-1\))的硬币数量,\(x\)为最终每个人的硬币数量.
这时候可能有的人就会疑惑了,为啥只给逆时针方向的人不给顺时针方向的人?
我们假设一下,\(1\)号给\(2\)号\(5\)个硬币,\(2\)号给\(1\)号\(3\)个硬币,一共转移了\(8\)个硬币,何必这么麻烦?直接让\(1\)号给\(2\)号\(2\)个硬币不就得了? 这样只需要转移\(2\)个硬币了.
注意,这个硬币数量可以是负数,也就是说,\(1\)号给\(6\)号\(-2\)个硬币,相当于6号给1号\(2\)个硬币
所以,相邻两人之间只会有一次硬币转移.
那我们擦去顺时针的:
然后就可以开始列式子了:
\(x=coin_1+pass_2-pass_1\)
\(x=coin_2+pass_3-pass_2\)
\(\dots\)
\(x=coin_n+pass_1-pass_n\)
移项
\(pass_2=x-coin_1+pass_1\)
\(pass_3=x-coin_2+\bm{pass_2}
\\ \ \ \ \ \ \ \ \ \ \ = x-coin_2+\bm{x_1-coin_1+pass_1}
\)
\(\dots\)
由题意,我们应该让\(\sum_{i=1}^n pass_i\)最小
为了方便描述,再设一个数组\(b_i\),使\(b_0=0\),\(b_i=coin_i+b_{i-1}-x\)
这样就可以方便的表示\(pass_i\),再表示一遍
\(pass_2=pass_1+x-coin_1
\\ \ \ \ \ \ \ \ \ \ \ = pass_1 - (coin_1-x)
\\ \ \ \ \ \ \ \ \ \ \ = pass_1 - b_1
\)
\(pass_3=\bm{pass_2}+x-coin_2
\\ \ \ \ \ \ \ \ \ \ \ = \bm{pass_1 - b_1} +x - coin_2
\\ \ \ \ \ \ \ \ \ \ \ = pass_1 - (\bm{coin_2+b_1-x})
\\ \ \ \ \ \ \ \ \ \ \ = pass_1 - \bm{b_2}
\)
\(\dots\)
所以
\(\sum_{i=1}^npass_i = pass_1+pass_2+pass_3+\dots+pass_n
\\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ = |pass_1 - b_1|+|pass_1 - b_2| + \dots + |pass_1 - b_{n-1}|
\)
注意,这个式子相当于数轴上\(b_1,b_2,b_3,b_4\cdots\)到\(pass_1\)的距离之和,那么使这个距离之和最小即可
这里我们可以就用中位数的性质了:
在数轴上所有n个点中,中位数离所有点的距离之和最小
求出最小的和输出即可
理解
对于这个数轴,中位数为2,与其他点距离的和为6
如果不选择中位数,比如选择1,那么与其他点距离的和为7
形象的理解,当\(pass_1\)向左移动一个单位长度时,虽然有两个点的距离缩小了\(d\),但是有3个点的距离增加了\(d\),显然结果变大
若移动两个单位长度,同理结果大于移动一个单位长度,所以当选择中位数时,结果最小.
如果出现偶数个点,
\([2,3]\)中任意一个点皆可做\(pass_1\),由于\(pass_1\)为整数并且为了写代码方便,取端点值即可
证明
给定一个从小到大的数列\(x_1,x_2,\dots x_n\)
设\(x\)是从\(x1\)到\(x_n\)与其绝对差之和最小的数.
当\(x\)在\(x_1\)与\(x_n\)之间,\(x\)与\(x_1\),\(x_n\)的距离和就是\(x_1\)到\(x_n\)的距离,
若不在\(x_1\)与\(x_n\)之间,\(x\)与\(x_1\),\(x_n\)的距离和还要额外加上一段,
则\(x\)位于\(x_1\)与\(x_n\)之间.由于\(x_1, x_n\)与它们之间的任意一点的距离之和(\(|x_i-x_1|+ |x_i- x_n\)|)都相等,等于\(x_n一x_1\),因此不需要考虑\(x_1, x_n\).
由第3点可得,\(x\)位于\(x_2\)与\(x_{n-1}\)之间,且\(x\)与\(x_2,x_{n-1}\)的距离和相等,不需要考虑.
依次类推,\(x\)就是该数列中间的那个数,或者是中间的那两个数之一, 而这个数就是中位数.
代码
#include <algorithm>
#include <cstdio>
using namespace std;
const int maxn = 1000000 + 10;
long long coin[maxn], b[maxn], tot, x;
int main() {
int n;
while (~scanf("%d", &n)) {
tot = b[0] = 0;
for (int i = 1; i <= n; i++) {
scanf("%lld", &coin[i]);
tot += coin[i];
}
x = tot / n;
for (int i = 1; i < n; i++) b[i] = coin[i] + b[i - 1] - x;
sort(b, b + n);
long long mid = b[n / 2], ans = 0;
for (int i = 0; i < n; i++) ans += abs(mid - b[i]);
printf("%lld\n", ans);
}
return 0;
}
UVA11300 Spreading the Wealth 题解的更多相关文章
- Uva11300 Spreading the Wealth
设第i个人需要给第i+1个人的金币数为xi(xi为负代表收到钱),列出一大堆方程. 设第i个人给第i-1个人的钱为xi(xi<0表示第i-1个人给第i个人钱).计算出最后每个人应该有的钱m,解方 ...
- 【题解】 UVa11300 Spreading the Wealth
题目大意 圆桌旁边坐着\(n\)个人,每个人有一定数量的金币,金币的总数能被\(n\)整除.每个人可以给他左右相邻的人一些金币,最终使得每个人的金币数量相等.您的任务是求出被转手的金币的数量的最小值. ...
- (洛谷P2512||bzoj1045) [HAOI2008]糖果传递 || 洛谷P4016 负载平衡问题 || UVA11300 Spreading the Wealth || (洛谷P3156||bzoj3293) [CQOI2011]分金币
bzoj1045 洛谷P4016 洛谷P2512 bzoj3293 洛谷P3156 题解:https://www.luogu.org/blog/LittleRewriter/solution-p251 ...
- UVA11300 Spreading the Wealth 数学
前方数学警告 题目链接:https://onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=25&am ...
- UVa 11300 Spreading the Wealth(有钱同使)
p.MsoNormal { margin: 0pt; margin-bottom: .0001pt; text-align: justify; font-family: "Times New ...
- UVA 11300 Spreading the Wealth (数学推导 中位数)
Spreading the Wealth Problem A Communist regime is trying to redistribute wealth in a village. They ...
- uva 11300 - Spreading the Wealth(数论)
题目链接:uva 11300 - Spreading the Wealth 题目大意:有n个人坐在圆桌旁,每个人有一定的金币,金币的总数可以被n整除,现在每个人可以给左右的人一些金币,使得每个人手上的 ...
- Uva 11300 Spreading the Wealth(递推,中位数)
Spreading the Wealth Problem A Communist regime is trying to redistribute wealth in a village. They ...
- Math - Uva 11300 Spreading the Wealth
Spreading the Wealth Problem's Link ---------------------------------------------------------------- ...
随机推荐
- (九)不安全的HTTP方法
01 漏洞描述 <HTTP | HTTP报文>中介绍了HTTP的请求方法.在这些方法中,HTTP定义了一组被称为安全方法的方法:GET.HEAD.OPTIONS.TRACE,这些方法不会产 ...
- Python 在线免费批量美颜,妈妈再也不用担心我 P 图两小时啦
引言 首先我承认自己标题党了,我就想提升点阅读量我容易么我,前几天的篇纯技术文阅读量都扯着蛋了. 毕竟阅读量太低实在是没有写下去的动力,我只能用点小手段偶尔提升下阅读量. 这篇文章我转换下套路,先放结 ...
- UniRx精讲(一):UniRx简介&定时功能实现
1.UniRx 简介 UniRx 是一个 Unity3D 的编程框架.它专注于解决时间上异步的逻辑,使得异步逻辑的实现更加简洁和优雅. 简洁优雅如何体现? 比如,实现一个"只处理第一次鼠标点 ...
- epic无法登陆怎么办【百分百解决】
epic无法登陆怎么办(感谢吾爱破解吧www.52pjb.net)站长提供的文章教程以及解决思路,谢谢大佬,更详细的解决办法可以参考下他的网站解决的方法实例! epic登陆不上去怎么办?随着GTA5免 ...
- 曹工改bug:centos下,mongodb开机不能自启动,systemctl、rc.local都试了,还是不行,要不要放弃?
问题背景 最近装个centos 7.6的环境,其中,基础环境包括,redis.nginx.mongodb.fastdfs.mysql等,其中,自启动使用的是systemctl,其他几个组件,都没啥问题 ...
- webpack介绍—上
6.1 webpack概念的引入 在网页中会引用哪些常见的静态资源? JS .js. .jsx ..coffee. .ts(TypeScript 类 C# 语言) CSS .css. .less. . ...
- 13.实战交付一套dubbo微服务到k8s集群(6)之交付dubbo服务的消费者集群到K8S
构建dubbo-demo-consumer,可以使用和dubbo-demo-service的流水线来构建 1.登录jenkins构建dubbo-demo-consumer 2.填写构建dubbo-d ...
- C#数据结构与算法系列(十):逆波兰计算器——逆波兰表达式(后缀表达式)
1.介绍 后缀表达式又称逆波兰表达式,与前缀表达式相似,只是运算符位于操作数之后 2.举例说明 (3+4)*5-6对应的后缀表达式就是3 4 +5 * 6 - 3.示例 输入一个逆波兰表达式(后缀表达 ...
- 微信小程序for循环遍历
wxml: <block wx:for="{{data}}" wx:for-item="data"> & ...
- Jmeter系列(34)- 详解 Counter 计数器
如果你想从头学习Jmeter,可以看看这个系列的文章哦 https://www.cnblogs.com/poloyy/category/1746599.html 简单介绍 计数器的作用:循环递增生成数 ...