题目链接

题意简介

将一个长度为 2n 的数列平均分为两个子数列 p 和 q 后,p 按从小到大排序,q 按从大到小排序。

排序后,记 p 为 \(\{x_i\}\) ,q 为 \(\{y_i\}\) ,对每种划分方式,有 \(f(p,q) = \sum |x_i - y_i|\)

现在我们想要知道对所有的划分方案 \((p,q)\) ,\(\sum f(p,q)\) 是多少。

数据范围:\(1 \leq n \leq 150000\) 答案对 998244353 取模。

Two partitions of an array are considered different if the sets of indices of elements included in the subsequence p are different.

这句话可以这么理解,就算元素的值相同,只要它们在原数列中的下标不同,就算为不同的元素。

只要原列组中有一个元素的所处位置( p 或 q )不同,就视为两种划分方式不同。

思路分析

考虑暴力,我们会发现我们共需要讨论 \({2n \choose n}\) 种情况,显然不能这么算。

(上面那个是组合数公式 2n 选 n)

于是我们自然而然地想到,既然对每种划分情况行不通,我们就考虑把每个数分开来,讨论其对于答案的贡献

通过对式子的观察,我们可以得出结论:\(x_i,y_i\) 中较大值对答案贡献为正,较小值对答案贡献为负

首先对原数列做排序处理。

现在我们对原数列进行从小到大排序,考虑从左到右选到第 i 个数 \(a_i\) 时,之前选了 \(k\) 个数在数列 q 中,\(i-1-k\) 个数在 p 中的情形。

(为避免重复计算与讨论的麻烦,不妨假设排序时,对于值相同的元素,在 \(\{a_i\}\) 中的下标越小越小。)

于是我们知道,前 i-1 个数都比 \(a_i\) 小。

由于我们从左到右选数,我们不难看出,每选到一个数加入数列 q,这个数将从右往左地添加到 q 中。而如果是加入数列 p,这个数将被从左往右地加入 p 中。如下图:

接下来我们分析,假定我们希望将 \(a_i\) 选入 p 中,那么 \(a_i\) 对应的实际上就是 \(x_{i-1-k}\),想要这个数对答案的贡献为,我们就需要使其对应的 \(y_{i-1-k}\) 比它小。由于我们已经将原数列排序,所以这个 \(y_{i-1-k}\) 在原数列中对应的 \(a_j\) 应有 \(j<i\) 。

而前面的数的选择我们实际上已经决定好了:我们选了 k 个数在 q 中。所以,我们必须要求这 k 个数中的某个数对于的 \(y_i\) 下标等于 \(i-1-k\) 。如下图。

显然,只有当 \(k\geq n-(i-k)+1\) 时,\(a_i\) 对答案的贡献为正。

经过简单的化简,我们得出 \(i > n\) 这样一条与 k 无关的式子。

换句话说,只要满足 \(i > n\) ,任何的将 \(a_i\) 放在 \(p\) 的情形,\(a_i\) 对答案的贡献都是正的。反之,贡献为负。

同理,假如我们考虑把 \(a_i\) 放到 q 中,同理,假如我们希望其贡献为正,那么 \(a_i\) 对应的 \(y_j\) 所对应的 \(x_j\) 所对应的 \(a_l\) 的下标应该出现在 i 之前,也就是比 \(a_i\) 小。

上面这句话可能有点绕。如下图。

显然,只有当 \(i-1-k \geq n-1-k+1\) 时,\(a_i\) 对答案的贡献为正。

化简后,我们又得到了同一条式子:\(i > n\) 。

于是,我们得出结论,无论怎么分,只要 \(i>n\) ,\(a_i\) 对答案的贡献就是正的,反之则是负的。

所以,答案就是 \({2n \choose n} \times (\sum_{i=n+1}^{2n} a_i - \sum_{i=1}^{n} a_i)\)

代码库

#include <cstdio>
#include <algorithm>
using namespace std;
typedef long long ll;
#define REG register
#define rep(i,a,b) for(REG int i=a;i<=b;i++)
const int N=3e5+5,mod=998244353;
int A[N],n; ll fac[N],ans1,ans2;
inline ll _pow(ll x,int p){
REG ll ans=1;
while(p) (p&1)&&(ans=ans*x%mod),x=x*x%mod,p>>=1;
return ans;
}
inline ll _inv(ll x){
return _pow(x,mod-2);
}
inline ll C(ll a,ll b){
return fac[a]*_inv(fac[b])%mod*_inv(fac[a-b])%mod;
}
int main(){
scanf("%d",&n);
rep(i,1,n<<1) scanf("%d",A+i);
sort(A+1,A+(n<<1)+1);
fac[0]=1;
rep(i,1,n<<1) fac[i]=fac[i-1]*i%mod;
ll temp=C(n<<1,n);
rep(i,1,n) ans1=(ans1+A[i]*temp)%mod;
rep(i,n+1,n<<1) ans2=(ans2+A[i]*temp)%mod;
printf("%lld\n",(-ans1+ans2+mod)%mod);
return 0;
}

【CF1445D】Divide and Sum 题解的更多相关文章

  1. Ural 1248 Sequence Sum 题解

    目录 Ural 1248 Sequence Sum 题解 题意 题解 程序 Ural 1248 Sequence Sum 题解 题意 给定\(n\)个用科学计数法表示的实数\((10^{-100}\s ...

  2. LeetCode Continuous Subarray Sum 题解 同余前缀和 Hash表

    文章目录 题意 思路 特殊情况k=0 Source Code 1 Source Code 2 题意 给定一个数组和一个整数k,返回是否存在一个长度至少为2的连续子数组的和为k的倍数. 思路 和上一篇博 ...

  3. Codeforces Round #680 (Div. 2, based on Moscow Team Olympiad) D. Divide and Sum (思维,数学,逆元)

    题意:有一个长度为\(2n\)数组,从中选分别选\(n\)个元素出来组成两个序列\(p\)和\(q\),(\(p\)和\(q\)中只要有任意一个元素在\(a\)的原位置不同,就算一个新的情况),选完后 ...

  4. Hdoj 1003.Max Sum 题解

    Problem Description Given a sequence a[1],a[2],a[3]......a[n], your job is to calculate the max sum ...

  5. [LeetCode]Combination Sum题解(DFS)

    Combination Sum Given a set of candidate numbers (C) (without duplicates) and a target number (T), f ...

  6. [LeetCode] Three Sum题解

    Three Sum: Given an array S of n integers, are there elements a, b, c in S such that a + b + c = 0? ...

  7. 01Two Sum题解

    Tow Sum 原题概述: Given an array of integers, return indices of the two numbers such that they add up to ...

  8. BZOJ3155:Preprefix sum——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=3155 最朴素的想法是两棵树状数组,一个记录前缀和,一个记录前缀前缀和,但是第二个我们非常不好修改 ...

  9. HDU4825:Xor Sum——题解

    http://acm.hdu.edu.cn/showproblem.php?pid=4825 Zeus 和 Prometheus 做了一个游戏,Prometheus 给 Zeus 一个集合,集合中包含 ...

随机推荐

  1. 微信App支付接入步骤&支付中前后端交互流程

    最近对微信App支付(App端集成微信支付SDK)申请步骤,以及终端在进行微信支付时商户App.商户Server.微信App.微信支付Server的交互流程进行了简单了解.这篇文章应该算是学习笔记,分 ...

  2. mysql CHAR and VARCHAR 比较

    写在前面 面试的时候突然有一位面试官问,说说CHAR和VARCHAR的区别,懵逼了,想自己平常使用的时候直接把VARCHAR拿来就用,真没注意到其中的不同. 反思,为什么没有注意到他们的不同 对于my ...

  3. python 递归求和

    例子: 1 mylist = [1,2,3,4,5] 2 def func(var): 3 if var == []: 4 print('列表为空') 5 return 0 6 return var[ ...

  4. CentOS 7的安装与部署 01

    01 虚拟软件的安装与配置 虚拟机(Virtual Machine)指通过软件模拟的具有完整硬件系统功能的.运行在一个完全隔离环境中的完整计算机系统.在实体计算机中能够完成的工作在虚拟机中都能够实现. ...

  5. 无法访问GitHub

    我们开发者经常用的最大的同性交流平台--GitHub忽然访问不了了,很尴尬 可以打开控制台 ping一下 github.com 果不其然 不通 不过幸运的是里面有github的ip地址,好像是美国某个 ...

  6. 第一期chrome浏览器的小技巧------《提高搜索效率》

    !!! 这次的技巧是:利用chrome提供的设置 提高你的搜索效率 !!! 我们经常遇到问题,搜索的时候很不方便 比如你在百度上搜索一个东西的时候正好没有搜到,那么你想找到这个东西的话,很明显要到其他 ...

  7. Jenkins:用maven在本地打包部署一个github的springboot项目(Jenkins 2.257)

    一,部署的准备工作 1,测试项目的地址: https://github.com/liuhongdi/profile 是一个空白项目 2,准备java环境:参见: https://www.cnblogs ...

  8. docker19.03使用数据卷

    一,查看当前的docker版本 [root@localhost data]# docker --version Docker version 19.03.8, build afacb8b 说明:刘宏缔 ...

  9. 第三十三章 linux常规练习题(二)

    一.练习题一 1.删除用户基本组shanghai03.发现无法正常删除,怎样才能将其删除掉,不能删除用户.2.打开多个xshell窗口连接登录同一虚拟机,使用不同的用户登录多次,分别使用w和who命令 ...

  10. Martyr2项目实现——Number部分问题求解(3) Prime Factorization

    Martyr2项目实现--Number部分问题求解(3) Prime Factorization 质因子分解 问题描述: Prime Factorization – Have the user ent ...