前方数学警告

题目链接:https://onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=25&page=show_problem&problem=2275

https://www.luogu.com.cn/problem/UVA11300

分析:

搞了这么久终于碰到一个数学题了,这也是本蒟蒻做的第一个数学题吧,值得纪念。

第一次看见这题的时候,嗯,好像是个暴力。。。然后就打了一个暴力,发现样例过了?但自己造了组数据就有了点问题,我暴力的思路是由于一个人只能给左边或右边,于是把这两种情况都枚举一下,取最小值就好,但有可能跑一遍不能让整个数组都变成我想要的那个值,故我又加了一个bool表示有无把所有数字都变成那个数,这么写肯定T啊。

我:万一数据水呢?

数据:我不水。

所以交上去果断T掉了,我又想验证一下这么写可不可以,找某人代码过来对拍了一下,然后结果是我竟然把一个A掉的代码给拍出了错解?嗯,看来写个暴力还是有收获的,我其实很想把那个代码粘上来的,但它88行有点长我没存不小心弄丢了,来看一下正解吧。

首先我最开始发现的那个性质是没有问题的,即每个人只会被它周围的人影响,一旦他周围的人的转移硬币数确定了,那么这个人的转移硬币数也可求,于是受这一点启发,不妨设\(x_i\)表示第\(i\)个人给第\(i-1\)个人的硬币数,当\(i==1\)时,为给\(n\)的硬币数。由于最后要求所有人金币一样(共产主义),所以最后每个人的金币数量是确定的,即\(M=tot/n\),也就是说每个人最后都是\(M\),于是有\(A_i+x_{i+1}-x_i=M\),就是原来有的加上得到的减去失去的等于最终的,光看这个式子,这啥也看不出来啊,所以继续观察这个式子,发现它跟周围的式子相加会出现一个神奇的结果,我们加一下看看。

\(A_1+x_2-x_1=M\) 1

\(A_2+x_3-x_2=M\) 2

\(A_3+x_4-x_2=M\) 3

1+2得\(A_1+A_2+x_3-x_1=M*2\) -> \(x_3=x_1-(A_1+A_2-M*2)\)

1+2+3得\(x_4=x_1-(A_1+A_2+A_3-M*3)\)

………………

不难归纳得出\(x_n=x_1-\)\(\sum_{i=1}^n(A_{i-1}+(i-1)*M\)\()\)

有这个式子我们就可以很简单困难的求出答案了,设\(B_i=\)\(\sum_{i=1}^n(A_{i-1}+(i-1)*M\),\(B\)数组可以由递推得到,那么问题就简化成了在数组\(B\)中求一个数,使得剩余数到它的距离之和最小,答案是中位数,证明如下。



假设\(a_1,a_2,a_3,a_4,a_5\)有序,则\(a_3\)是中位数无疑,我们用线段来表示点与点之间的距离,可以看出蓝色框框内的线段是必不可少的,不管怎么划分都必须要有,如果这个点在\(a_3\)的左边,那么就会多出粉色的一截,在右边就会多出红色的一截,所以在中位数时是最小的。

那偶数呢?偶数不是有两个中位数吗?偶数时也画出一个这样的图,会发现当点在两个中位数之间距离都是一样的,并且是最小,所以为了方便,每次中位数都取\(n+1>>1\)就可以了。

中位数怎么找呢,排序一遍可以,但复杂度是\(O(nlogn)\)的,如果要使数组有序,那么肯定是要\(sort\)的,这里只求中位数,所以可以用\(nth\)\(element\),这个函数的用法是\(nth\)\(element(A+a,b,A+c)\),表示将数组A的区间\([a,c)\)中第b大的数放在b位置上,不保证数组有序,但比这个数大的都在它前边,比它小的在它后边,平均时间复杂度\(O(n)\)。

#include<iostream>
#include<cmath>
#include<algorithm>
#define ll long long
using namespace std;
const int N=1e6+10;
ll a[N],b[N];
int main(){
ios::sync_with_stdio(false);
int n;
while(cin>>n){
ll tot=0;
for(int i=1;i<=n;i++)
cin>>a[i],tot+=a[i],b[i]=0;
tot/=n;
for(int i=2;i<=n;i++)
b[i]=b[i-1]+a[i]-tot;
int mid=n+1>>1;
nth_element(b+1,b+mid,b+n+1);
ll temp=b[mid],res=0;
for(int i=1;i<=n;i++)
res+=abs(temp-b[i]);
cout<<res<<'\n';
}
}

tips:cout<<endl的确比cout<<'\n'慢一点。

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. UVA11300 Spreading the Wealth 题解

    题目 A Communist regime is trying to redistribute wealth in a village. They have have decided to sit e ...

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

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

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

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

  6. 数学/思维 UVA 11300 Spreading the Wealth

    题目传送门 /* 假设x1为1号给n号的金币数(逆时针),下面类似 a[1] - x1 + x2 = m(平均数) 得x2 = x1 + m - a[1] = x1 - c1; //规定c1 = a[ ...

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

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

  8. Math - Uva 11300 Spreading the Wealth

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

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

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

随机推荐

  1. Matplotlib数据可视化(5):柱状图与直方图

      柱状图和直方图是两种非常类似的统计图,区别在于: 直方图展示数据的分布,柱状图比较数据的大小. 直方图X轴为定量数据,柱状图X轴为分类数据.因此,直方图上的每个条形都是不可移动的,X轴上的区间是连 ...

  2. vue安卓4.4.2页面打不开的坑

    项目上线两三天,有保障说安卓下面页面打不开,所以查了下具体原因,系统版本过低,安卓4.4.2,然后发现本地没有babel-polyfill的包,具体解决方案如下: 1,npm 安装 npm insta ...

  3. 最简单的???ubuntu 通过crontab定时执行一个程序

    crontab在liunx系统中下载,我默认是认为下载安装了的.. crontab貌似只能在liunx系统中存在,如果是windows系统我不知道 创建一个名为jiaoben的文件夹存储sh文件,进入 ...

  4. day1 执行用例

    故事1 今天的工作,是执行已写好的"回馈"模块用例 每个子模块的用例都有优先级之分 肯定要先进行通过性测试,验证正常流 因此,我优先执行各个子模块优先级最高的那条用例,进行通过性测 ...

  5. 用shell脚本实现定时备份数据库

    1.备份数据库的方法 可以使用命令查看 ls  /usr/local/mysql/bin 这个mysqldump就是系统内置的用来备份数据库的工具. 2.实现方法 ①先随便进入一个位置创建一个目录 ② ...

  6. 鸡汤 & 毒鸡汤

    1.别低估任何人. 2.你没那么多观众,别那么累. 3.温和对人对事.不要随意发脾气,谁都不欠你的. 4.现在很痛苦,等过阵子回头看看,会发现其实那都不算事. 5.和对自己有恶意的人绝交.人有绝交,才 ...

  7. 5G 将带给程序员哪些新机会呢?

    5G,第 5 代移动通信技术,华为在此领域远远领先同行,这也让它成了中美贸易战的最前线.我的第一份工作就在通信行业,当时电信标准都在欧美企业手里,国内企业主要是遵照标准研发软硬件设备,核心芯片靠进口. ...

  8. tomcat启动时间5分钟左右org.apache.catalina.util.SessionIdGeneratorBase.createSecureRandom Creation of SecureRandom instance for session ID generation using [SHA1PRNG] took [342,445] milliseconds.

    org.apache.catalina.util.SessionIdGeneratorBase.createSecureRandom Creation of SecureRandom instance ...

  9. P1969 积木大赛 题解

    原题链接 简要题意: 每次把一段区间 \(+1\),问得到 \(a\) 数组的最小次数. 我们可以把 \(+1\) 得到 \(a\) 换成,从 \(a\) 依次 \(-1\) 得到 \(0\). 算法 ...

  10. WordPress 版本升级、主题升级记录

    版本升级 升级很简单,但是以防万一,先备份数据. 一.备份数据库 mysqldump -u root -p --database myblog > myblog.sql 若需要还原可执行如下操作 ...