题目链接

题意简介

将一个长度为 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. oracle数据库外部连接无法访问

    服务器出现的问题是运行的项目无法访问oracle数据库连接,用plsql输入用户名密码后卡死,无法连接.但是通过命令窗口对oracle数据库操作正常,对oracle服务进行查看并重启,并无异常,运行t ...

  2. OpenCV计算机视觉学习(2)——图像算术运算 & 掩膜mask操作(数值计算,图像融合,边界填充)

    在OpenCV中我们经常会遇到一个名字:Mask(掩膜).很多函数都使用到它,那么这个Mask到底是什么呢,下面我们从图像基本运算开始,一步一步学习掩膜. 1,图像算术运算 图像的算术运算有很多种,比 ...

  3. C/C++ typedef用法

    原文来源:https://blog.csdn.net/superhoy/article/details/53504472 第一.四个用途 用途一: 定义一种类型的别名,而不只是简单的宏替换.可以用作同 ...

  4. P5858 「SWTR-03」Golden Sword

    题面: Link 题面有点长,不想粘了,QAQ. 题解: 一句话题意,你有 \(n\) 件物品需要依次放进去,每个物品放进去之后会得到一定的权值,为当前锅炉里面的物品的数量乘以 \(a_i\) 每次在 ...

  5. Activity常用方法

    setContentView(r.layout.xxxx);//设置布局文件 getViewById(r.id.xxxx);//获取指定控件 getString(r.string.xxxx);//获取 ...

  6. Jmeter JDBC Request 使用详解

    本篇博文讲解以MySQL为例,搞懂JDBC Request中MySQL的使用方法,换成其它数据库, 如Oracle.PSQL也会很容易上手. 一.基本配置 1.首先我们先了解一下,不同数据库的驱动类和 ...

  7. 【Flutter Widgets大全】电子书开源

    [Flutter Widgets大全]是老孟耗费大量精力整理的,总共有330多个组件的详细用法,开源到Github上,希望可以帮助到大家,开源不易,点个赞可不可以. [Flutter Widgets ...

  8. 每日一题 LeetCode 491. 递增子序列 【递推】【递增子序列】【动态规划】

    题目链接 https://leetcode-cn.com/problems/increasing-subsequences/ 题目说明 题解 主要方法:递推:动态规划 解释说明: 数据表示:观察数据范 ...

  9. NET::ERR_INCOMPLETE_CHUNKED_ENCODING 200 (OK)

    错误信息: NET::ERR_INCOMPLETE_CHUNKED_ENCODING 200 (OK) 错误背景:微服务不通过统一的nginx端口访问,能够正常请求接口并获取对应的响应.但是通过ngi ...

  10. 洛谷 CF1012C Hills(动态规划)

    题目大意: 有几座山,如果一座山左右两边的山比它矮,那么可以在这个山上建房子,你有一台挖掘机,每天可以挖一座山一米,问你需要花多少代价可以分别盖1.2.3--座房子.(给出山的数量,以及每座山的高度) ...