题解-CSA Beta Round#1 Number Elimination
Problem
题意概要:给定 \(n\) 个数组成的序列,定义一次操作:
- 在当前序列中选择两个数,将其中较小的数从序列中删除(若两个数相同,则删除在序列中更靠前的)
- 较大的数为此次操作的花费
进行 \(n-1\) 次操作后只剩一个数,求有多少种不同的执行方案,满足其花费为所有方案中花费最少的
Solution
发现网上没有什么CSA方面的资料?
首先有一个贪心,就是将数字排序离散后压缩一下,若数字 \(i\) 有 \(s_i\) 个,则一定先来 \(s_i-1\) 次花费为零的操作(每次选择两个 \(i\)),剩下的一个 \(i\) 一定是被一个 \(i+1\) 删除
然后考虑用操作的拓扑图来统计……没辙,正解是序列 \(dp\)
先设 \(dp[i]\) 表示 \(i\) 个相同的数字互相消除直至只剩一个数字的方案数,易得 \(dp[n] = \prod_{i=2}^n\frac {i(i-1)}2\)
再考虑设 \(f[i]\) 为前 \(i\) 堆数字已经消成只剩一个的方案数
考虑在处理完前 \(i-1\) 堆的基础上加入第 \(i\) 堆,不妨枚举在第 \(i-1\) 堆的最后一个元素被消除之前有多少个 \(i\) 被消除了,易得:
\]
一个个解释:
- \(f[i-1]\):前 \(i-1\) 堆消成一个 \(i-1\) 的方案数
- \(dp[s_i]\):这 \(s_i\) 个 \(i\) 消成一个的方案数
- \(s_i-x\):前 \(i-1\) 堆消剩下的那一个 \(i-1\) 需要从剩下的 \(s_i-x\) 个 \(i\) 中挑一个消它
- \(\binom {\sum_{j=1}^{i-1}s_i-1+x}x\):考虑将这 \(x\) 个提前消的融入前面的消除序列(这里不使用排列是因为这 \(x\) 个 \(i\) 内部之间的前后关系已经在 \(dp[s_i]\) 中考虑过了)
总的来说,就是将 \(s_i\) 分为两段,一份放入前面一起大排队,另一份用来消剩下的那个 \(i-1\)
复杂度 \(O(\sum s_i)=O(n)\)
Code
#include <bits/stdc++.h>
typedef long long ll;
inline void read(int&x){
char ch=getchar();x=0;while(!isdigit(ch))ch=getchar();
while(isdigit(ch))x=x*10+ch-'0',ch=getchar();
}
const int N = 101000, p = 1e9+7;
int a[N], sum[N], f[N], dp[N], n;
inline int qpow(int A, int B) {
int res = 1; while(B) {
if(B&1) res = (ll)res * A%p;
A = (ll)A * A%p, B >>= 1;
} return res;
}
int fac[N], inv[N];
inline int C(const int nn, const int mm) {return (ll)fac[nn] * inv[mm]%p * inv[nn-mm]%p;}
int main() {
read(n);
for(int i=1;i<=n;++i) read(a[i]);
std::sort(a+1,a+n+1);
dp[0] = dp[1] = fac[0] = fac[1] = 1;
for(int i=2;i<=n;++i) {
fac[i] = (ll)fac[i-1] * i%p;
dp[i] = ((ll)i * (i-1) >> 1) * dp[i-1]%p;
}
inv[n] = qpow(fac[n], p-2);
for(int i=n;i;--i) inv[i-1] = (ll)inv[i] * i%p;
int t = 0;
for(int i=1,j=1;i<=n;i=j) {
while(a[i] == a[j] and j <= n) ++j;
a[++t] = j - i, sum[t] = (a[t] + sum[t-1])%p;
}
n = t;
f[1] = dp[a[1]];
for(int i=2;i<=n;++i) {
for(int x = 0; x < a[i]; ++ x)
f[i] = (f[i] + (ll)C(sum[i-1]-1+x,x) * (a[i]-x))%p;
f[i] = (ll)f[i] * f[i-1]%p * dp[a[i]]%p;
}
printf("%d\n", f[n]);
return 0;
}
题解-CSA Beta Round#1 Number Elimination的更多相关文章
- Codeforces Beta Round #62 题解【ABCD】
Codeforces Beta Round #62 A Irrational problem 题意 f(x) = x mod p1 mod p2 mod p3 mod p4 问你[a,b]中有多少个数 ...
- Codeforces Beta Round #83 (Div. 1 Only)题解【ABCD】
Codeforces Beta Round #83 (Div. 1 Only) A. Dorm Water Supply 题意 给你一个n点m边的图,保证每个点的入度和出度最多为1 如果这个点入度为0 ...
- Codeforces Beta Round #80 (Div. 2 Only)【ABCD】
Codeforces Beta Round #80 (Div. 2 Only) A Blackjack1 题意 一共52张扑克,A代表1或者11,2-10表示自己的数字,其他都表示10 现在你已经有一 ...
- Codeforces Beta Round #5 B. Center Alignment 模拟题
B. Center Alignment 题目连接: http://www.codeforces.com/contest/5/problem/B Description Almost every tex ...
- Codeforces Beta Round 84 (Div. 2 Only)
layout: post title: Codeforces Beta Round 84 (Div. 2 Only) author: "luowentaoaa" catalog: ...
- Codeforces Beta Round #17 D. Notepad (数论 + 广义欧拉定理降幂)
Codeforces Beta Round #17 题目链接:点击我打开题目链接 大概题意: 给你 \(b\),\(n\),\(c\). 让你求:\((b)^{n-1}*(b-1)\%c\). \(2 ...
- Codeforces Beta Round #16 E. Fish (状压dp)(概率dp)
Codeforces Beta Round #16 (Div. 2 Only) E. Fish 题目链接:## 点击打开链接 题意: 有 \(n\) 条鱼,每两条鱼相遇都会有其中一只吃掉对方,现在给你 ...
- Codeforces Beta Round #13 C. Sequence (DP)
题目大意 给一个数列,长度不超过 5000,每次可以将其中的一个数加 1 或者减 1,问,最少需要多少次操作,才能使得这个数列单调不降 数列中每个数为 -109-109 中的一个数 做法分析 先这样考 ...
- Codeforces Beta Round #79 (Div. 2 Only)
Codeforces Beta Round #79 (Div. 2 Only) http://codeforces.com/contest/102 A #include<bits/stdc++. ...
随机推荐
- Java开发手册-编程规约精选
# Java开发手册-编程规约精选 ## 总约 - 采用驼峰写法 ## 变量 - 首字母小写 ## 方法 - 方法名首字母小写- 参数首字母小写 ## 引用 - <阿里巴巴Java开发手册> ...
- linux下apache安装ssl步骤
制作证书: 参考:linux下运用opensll制作ssl证书 生成三个证书 server.crt .server-ca.crt.server.key 安装openssl tar -xzvf open ...
- 003-结构型-07-享元模式(Flyweight)
一.概述 提供了减少对象数且从而改善应用所需的对象结构的方式.运用共享技术有效地支持大是细粒度的对象. 它通过与其他类似对象共享数据来减小内存占用.它使用共享物件,用来尽可能减少内存使用量以及分享资讯 ...
- 一些有用的github收藏(持续更新中...)
1.facebook的c++开源库folly(Facebook open source library)介绍 https://github.com/facebook/folly 2.pprint 一个 ...
- Spring cloud微服务安全实战-6-9sentinel之熔断降级
来讲一下降级规则 服务会互相调用,服务A会有一些服务之间的依赖. 假设服务D的响应时间变长了.A调用D就会卡住了. 熔断,某一个服务出现问题,会把服务拖死.如果A出现,会把依赖A的那些服务拖死. 主要 ...
- 中标麒麟+Qt+mysql解决驱动无法加载的问题
问题描述:都安装了Qt,Mysql之后,发现Qt始终不能连接Mysql 1.安装Qt 2.写程序直接连接QMysql 打印QSqlDatabase: * driver not loaded ,进入/h ...
- RabbitMQ 入门教程(PHP版) 简单Demo
RabbitMQ的关键字说明 (1)Broker:经纪人.提供一种传输服务,维护一条从生产者到消费者的传输线路,保证消息数据能按照指定的方式传输.粗略的可以将图中的RabbitMQ Server当作B ...
- 一些Python中的二维数组的操作方法
一些Python中的二维数组的操作方法 这篇文章主要介绍了一些Python中的二维数组的操作方法,是Python学习当中的基础知识,需要的朋友可以参考下 需要在程序中使用二维数组,网上找到一种这样的用 ...
- Ubuntu18使用netplan设置多网口绑定
Ubuntu18使用netplan设置网络参考:https://www.cnblogs.com/minseo/p/11325384.html 修改配置文件 /etc/netplan/50-cloud- ...
- 【ARTS】01_42_左耳听风-201900826~201900901
ARTS: Algrothm: leetcode算法题目 Review: 阅读并且点评一篇英文技术文章 Tip/Techni: 学习一个技术技巧 Share: 分享一篇有观点和思考的技术文章 Algo ...