题目

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\)为整数并且为了写代码方便,取端点值即可

证明

  1. 给定一个从小到大的数列\(x_1,x_2,\dots x_n\)

  2. 设\(x\)是从\(x1\)到\(x_n\)与其绝对差之和最小的数.

  3. 当\(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\)之间.

  4. 由于\(x_1, x_n\)与它们之间的任意一点的距离之和(\(|x_i-x_1|+ |x_i- x_n\)|)都相等,等于\(x_n一x_1\),因此不需要考虑\(x_1, x_n\).

  5. 由第3点可得,\(x\)位于\(x_2\)与\(x_{n-1}\)之间,且\(x\)与\(x_2,x_{n-1}\)的距离和相等,不需要考虑.

  6. 依次类推,\(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 题解的更多相关文章

  1. Uva11300 Spreading the Wealth

    设第i个人需要给第i+1个人的金币数为xi(xi为负代表收到钱),列出一大堆方程. 设第i个人给第i-1个人的钱为xi(xi<0表示第i-1个人给第i个人钱).计算出最后每个人应该有的钱m,解方 ...

  2. 【题解】 UVa11300 Spreading the Wealth

    题目大意 圆桌旁边坐着\(n\)个人,每个人有一定数量的金币,金币的总数能被\(n\)整除.每个人可以给他左右相邻的人一些金币,最终使得每个人的金币数量相等.您的任务是求出被转手的金币的数量的最小值. ...

  3. (洛谷P2512||bzoj1045) [HAOI2008]糖果传递 || 洛谷P4016 负载平衡问题 || UVA11300 Spreading the Wealth || (洛谷P3156||bzoj3293) [CQOI2011]分金币

    bzoj1045 洛谷P4016 洛谷P2512 bzoj3293 洛谷P3156 题解:https://www.luogu.org/blog/LittleRewriter/solution-p251 ...

  4. UVA11300 Spreading the Wealth 数学

    前方数学警告 题目链接:https://onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=25&am ...

  5. UVa 11300 Spreading the Wealth(有钱同使)

    p.MsoNormal { margin: 0pt; margin-bottom: .0001pt; text-align: justify; font-family: "Times New ...

  6. UVA 11300 Spreading the Wealth (数学推导 中位数)

    Spreading the Wealth Problem A Communist regime is trying to redistribute wealth in a village. They ...

  7. uva 11300 - Spreading the Wealth(数论)

    题目链接:uva 11300 - Spreading the Wealth 题目大意:有n个人坐在圆桌旁,每个人有一定的金币,金币的总数可以被n整除,现在每个人可以给左右的人一些金币,使得每个人手上的 ...

  8. Uva 11300 Spreading the Wealth(递推,中位数)

    Spreading the Wealth Problem A Communist regime is trying to redistribute wealth in a village. They ...

  9. Math - Uva 11300 Spreading the Wealth

    Spreading the Wealth Problem's Link ---------------------------------------------------------------- ...

随机推荐

  1. Linux 文件系统常用命令

    文件系统查看命令df df:查看分区,单位默认是KB df -h 统计目录或文件大小du du /etc/:会列出/etc/目录下的所有子目录所占的空间,最后给出/etc/目录的大小,属于高负载命令, ...

  2. zabbix 监控进程,端口

    环境介绍 操作系统:centos 7.4 zabbix版本:zabbix server 3.4.7 客户端:zabbix-agent 3.4.7 监控进程:mysqld 监控端口:3306 tcp 进 ...

  3. 01-Python初体验

    本节内容 Python介绍 发展史 Python 2 or 3? 安装 Hello World程序 变量 用户输入 模块初识 .pyc是个什么鬼? 数据类型初识 数据运算 表达式if ...else语 ...

  4. js循环练习

    var a=1; while(a>0){ var b=prompt('input number'); if(b>a){ alert('big'); } else if(b<a){ a ...

  5. curlPost和curlGet 请求链接

    //getcurl get读取数据function curlGet($url){ $UserAgent = 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) App ...

  6. ca70a_c++_重载函数_实参类型转换

    /*ca70a_c++_重载函数_实参类型转换转换等级,详见P290 编译选择哪个函数版本呢?1.精确匹配2.通过类型提升3.通过标准转换4.通过类类型转换参数匹配和枚举类型重载和const形参 vo ...

  7. JAVA 经典算法 40 例

    [程序 1] 题目:古典问题:有一对兔子,从出生后第 3 个月起每个月都生一对兔子,小兔子长到第四个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? 1.程序分析: 兔子的规律为数列 ...

  8. 完美解决asp.net core 3.1 两个AuthenticationScheme(cookie,jwt)共存在一个项目中

    内容 在我的项目中有mvc controller(view 和 razor Page)同时也有webapi,那么就需要网站同时支持2种认证方式,web页面的需要传统的cookie认证,webapi则需 ...

  9. (二)、JAVA运行时数据区域

    根据<Java 虚拟机规范(Java SE 7版)>规定,Java虚拟机所管理的内存,将会包括以下几个运行时数据区域: 注: 1.由所有线程共享的数据区: 对应 java内存模型的主内存, ...

  10. C#数据结构与算法系列(十四):递归——八皇后问题(回溯算法)

    1.介绍 八皇后问题,是一个古老而著名的问题,是回溯算法的经典案例,该问题是国际西洋棋棋手马克斯.贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即 任意两个皇后都不能处 ...