ARC120F Wine Thief (组合数学)
题面
有一个长为
N
N
N 的序列,相邻的两个数中只能选一个,总共选
k
k
k 个数,一种方案的价值为选的
k
k
k 个数的和,问所有合法方案的价值总和,答案对 998244353
取模。
1
≤
N
≤
3
⋅
1
0
5
,
1
≤
k
≤
⌈
N
2
⌉
1\leq N\leq 3\cdot10^5~,~1\leq k\leq \left\lceil \frac{N}{2} \right\rceil
1≤N≤3⋅105 , 1≤k≤⌈2N⌉.
题解
把每个数的贡献拆开求,答案就是 每个数的值 × 该数被选的方案数。这是最基本的转化。
令
G
(
n
,
i
,
k
)
G(n,i,k)
G(n,i,k) 表示
n
n
n 个数中选
k
k
k 个数,必选第
i
i
i 个数的方案数。
我们先鲁莽地列个式子,枚举第
i
i
i 个数前面选了多少个数,然后就是个放球问题,不难发现
G
(
n
,
i
,
k
)
=
∑
j
=
0
(
i
−
1
)
/
2
C
i
−
1
−
j
j
⋅
C
n
−
i
−
k
−
1
+
j
k
−
1
−
j
G(n,i,k)=\sum_{j=0}^{(i-1)/2}C_{i-1-j}^{j}\cdot C_{n-i-k-1+j}^{k-1-j}
G(n,i,k)=∑j=0(i−1)/2Ci−1−jj⋅Cn−i−k−1+jk−1−j
——毫无可做性。
我们得换个思路想。不妨容斥一下?先设
f
(
n
,
k
)
f(n,k)
f(n,k) 为在
n
n
n 个数中选
k
k
k 个数的总方案数,可得
f
(
n
,
k
)
=
C
n
−
k
+
1
k
f(n,k)=C_{n-k+1}^{k}
f(n,k)=Cn−k+1k。
然后这里利用了官解里所说的 “ 化环 ” 的思想。
如果把整个序列视为一个环(即第一个点和最后一个点不能同时选),那么
G
′
(
n
,
1
,
k
)
=
G
′
(
n
,
2
,
k
)
=
⋯
=
G
′
(
n
,
n
,
k
)
G'(n,1,k)=G'(n,2,k)=\cdots=G'(n,n,k)
G′(n,1,k)=G′(n,2,k)=⋯=G′(n,n,k),不妨令此时的
G
′
(
n
,
1
,
k
)
=
G
′
(
n
,
2
,
k
)
=
⋯
=
G
′
(
n
,
n
,
k
)
=
F
(
n
,
k
)
G'(n,1,k)=G'(n,2,k)=\cdots=G'(n,n,k)~=~F(n,k)
G′(n,1,k)=G′(n,2,k)=⋯=G′(n,n,k) = F(n,k),则通过一番组合推导,可以发现
F
(
n
,
k
)
=
{
n
<
3
:
[
k
=
=
1
]
n
≥
3
:
f
(
n
−
3
,
k
−
1
)
F(n,k)=\bigg\lbrace{\begin{matrix}n<3:~~[k==1]~~~~~~~~~\\ n\geq3:f(n-3,k-1) \end{matrix}}
F(n,k)={n<3: [k==1] n≥3:f(n−3,k−1)
如果不是个环,那就多了一种情况:第一个点和最后一个点同时选,我们把它加上去就行了。选了第一个点和最后一个点后,相当于中间的长度为
n
−
4
n-4
n−4 的子区间又构成了一个子问题,因此
G
(
n
,
i
,
k
)
G(n,i,k)
G(n,i,k) 有这样的递推公式:
G
(
n
,
i
,
k
)
=
{
i
≤
0
:
0
i
=
1
:
F
(
n
,
k
)
+
f
(
n
−
4
,
k
−
2
)
i
>
1
:
F
(
n
,
k
)
+
G
(
n
−
4
,
i
−
2
,
k
−
2
)
i
>
⌈
n
2
⌉
:
G
(
n
,
n
−
i
+
1
,
k
)
G(n,i,k)=\begin{cases} i\leq0: & 0\\ i=1: & F(n,k)+f(n-4,k-2)\\ i>1: & F(n,k)+G(n-4,i-2,k-2)\\ i>\lceil\frac{n}{2}\rceil: & G(n,n-i+1,k) \end{cases}
G(n,i,k)=⎩⎪⎪⎪⎨⎪⎪⎪⎧i≤0:i=1:i>1:i>⌈2n⌉:0F(n,k)+f(n−4,k−2)F(n,k)+G(n−4,i−2,k−2)G(n,n−i+1,k)
我们将一般情况(
i
>
1
i>1
i>1)展开,可以得到:
G
(
n
,
i
,
k
)
=
F
(
n
,
k
)
+
F
(
n
−
4
,
k
−
2
)
+
F
(
n
−
8
,
k
−
4
)
+
⋯
G(n,i,k)=F(n,k)+F(n-4,k-2)+F(n-8,k-4)+\cdots
G(n,i,k)=F(n,k)+F(n−4,k−2)+F(n−8,k−4)+⋯
不难发现对于不同的
i
i
i ,前面部分都是一样的,
i
i
i 只决定式子长度以及最后一项!
那么我们就可以从左到右遍历
i
i
i 的时候把
G
(
n
,
i
,
k
)
G(n,i,k)
G(n,i,k) 衔接着求,只求前一半。每次到新的
i
i
i 最多会增加或减少一两项。由于递推式中每次
i
i
i 会减少 2,且
i
=
1
i=1
i=1 属于特殊情况,所以要分奇偶性讨论。
具体实现有点细节,但是只求前一半,再考虑考虑边界应该就没问题了。
时间复杂度
O
(
n
)
O(n)
O(n)。
CODE
#include<cstdio>
#include<vector>
#include<cmath>
#include<queue>
#include<map>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 1000005
#define DB double
#define LL long long
#define ENDL putchar('\n')
#define lowbit(x) ((-x) & (x))
#define INF 0x3f3f3f3f
LL read() {
LL f=1,x=0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
const int MOD = 998244353;
int n,m,i,j,s,o,k;
int a[MAXN];
int fac[MAXN],inv[MAXN],invf[MAXN];
inline int qkpow(int a,int b) {
int res = 1;
while(b > 0) {
if(b & 1) res = res *1ll* a % MOD;
a = a *1ll* a % MOD; b >>= 1;
}return res;
}
inline int C(int n,int m) {
if(m < 0 || m > n) return 0;
return 1ll*fac[n] * invf[n-m] % MOD *invf[m] % MOD;
}
inline int f_(int n,int k) {return C(n-k+1,k);}
inline int F(int n,int k) {
if(n <= 3) {
if(k == 1) return 1;
else return 0;
}
else return f_(n-3,k-1);
}
int tm[MAXN];
int main() {
n = read();k = read();int D = read();
int ans = 0;
fac[0] = fac[1] = inv[0] = inv[1] = invf[1] = invf[0] = 1;
for(int i = 2;i <= n;i ++) {
fac[i] = fac[i-1] *1ll* i % MOD;
inv[i] = (MOD-inv[MOD%i]) *1ll* (MOD/i) % MOD;
invf[i] = invf[i-1] *1ll* inv[i] % MOD;
}
for(int i = 1;i <= n;i ++) {
a[i] = read(); tm[i] = -1;
}
int ans0 = 0,ans1 = f_(n,k),ad1 = 0,ad2 = 0;
for(int i = 1;i <= n;i ++) {
if(i & 1) {
(ans1 += MOD-f_(n-4*ad1,k-2*ad1)) %= MOD;
(ans1 += F(n-4*ad1,k-2*ad1)) %= MOD;
ad1 ++;
(ans1 += f_(n-4*ad1,k-2*ad1)) %= MOD;
if(tm[i] >= 0) ans1 = tm[i];
else tm[i] = tm[n-i+1] = ans1;
(ans += ans1 *1ll* a[i] % MOD) %= MOD;
}
else {
(ans0 += F(n-4*ad2,k-2*ad2)) %= MOD; ad2 ++;
if(tm[i] >= 0) ans0 = tm[i];
else tm[i] = tm[n-i+1] = ans0;
(ans += ans0 *1ll* a[i] % MOD) %= MOD;
}
}
printf("%d\n",ans);
return 0;
}
ARC120F Wine Thief (组合数学)的更多相关文章
- 17995 Stupid thief 组合数学
17995 Stupid thief 时间限制:1000MS 内存限制:65535K提交次数:0 通过次数:0 题型: 编程题 语言: 不限定 Description A stupid thie ...
- 记录在linux下的wine生活
记录在linux下的windows生活 本篇内容涉及QQ.微信.Office的安装配置 QQ: 到deepin下载轻聊版. 如果安装了crossover,那么将其中opt/cxoffice/suppo ...
- linux Mint wine安装qq,桌面快捷键配置
在桌面新建qq.desktop文件,添加以下内容 #!/usr/bin/env xdg-open[Desktop Entry]Comment=QQTerminal=falseName=QQExec=w ...
- Wine——在Linux上运行Windows软件
官网:https://www.winehq.org/ 参考: wikipedia 教你使用Wine在Linux上运行Windows软件 如何安装和使用Wine,以便在Linux上运行Windows应用 ...
- codeforces 632+ E. Thief in a Shop
E. Thief in a Shop time limit per test 5 seconds memory limit per test 512 megabytes input standard ...
- linux安装wine
1.添加PPA sudo add-apt-repository ppa:ubuntu-wine/ppa 2.更新列表 sudo apt-get update 3.安装Wine sudo apt-get ...
- Codeforces632E Thief in a Shop(NTT + 快速幂)
题目 Source http://codeforces.com/contest/632/problem/E Description A thief made his way to a shop. As ...
- CentOS 安装 Wine
1. 下载安装包 Wine的中文官网可以下载到最新稳定和开发版本的Wine安装包,根据不同需求可以自行下载 2. 解压安装包,编译前检查 根据不同的平台选择不同的编译选项: For 32-Bit Sy ...
- wine
sudo dpkg --add-architecture i386 sudo add-apt-repository ppa:wine/wine-buildssudo apt-get update su ...
随机推荐
- 【Redis】事件驱动框架源码分析(多线程)
IO线程初始化 Redis在6.0版本中引入了多线程,提高IO请求处理效率. 在Redis Server启动函数main(server.c文件)中初始化服务之后,又调用了InitServerLast函 ...
- 你真的很了解printf函数吗?
对C语言中经常使用的printf这个库函数,你是否真的吃透了呢? 系统化的学习C语言程序设计,是不是看过一两本C语言方面的经典著作就足够了呢?答案是显而易见的:不够.通过这种典型的入门级的学习方式,是 ...
- vue大型电商项目尚品汇(后台篇)day05
今天继续是对后台管理部分的一个操作,但是快要结束了,今天结束,明天会进入一个从Vue以来,另外一个名声显著的东西了,一只耳闻从未见识,而且十分的炫酷 他就是------数据可视化Echarts,迫不及 ...
- Vue3.0系列——「vue3.0学习手册」第一期
一.项目搭建 vite是尤大大开发的一款意图取代webpack的工具.其实现原理是利用ES6的import发送请求加载文件的特性.拦截这些请求,做一些编译,省去webpack冗长的打包时间.并将其与R ...
- MongoDB 的内存使用限制
本文将简述一下MongoDB的内存限制问题 1. 使用Docker限制 当我们使用docker创建mongo 容器时,可通过使用以下参数,对mongo可以使用的资源进行限制 内存限制 参数 简介 -m ...
- VirtualBox虚拟机安装Ubuntu系统后,增加内存空间和处理器核心数
对于Linux爱好者而言,初次使用虚拟机时,一般都会使用默认的设置,例如硬盘空间.内存空间等等. 而往往在熟悉之后,安装了某些必要的软件,或者熟悉了实际的开发场景后,却发现原本给虚拟机分配的物理资源是 ...
- Python实现12种概率分布(附代码)
今天给大家带来的这篇文章是关于机器学习的,机器学习有其独特的数学基础,我们用微积分来处理变化无限小的函数,并计算它们的变化:我们使用线性代数来处理计算过程:我们还用概率论与统计学建模不确定性. 在这其 ...
- 《AlignedReID:Surpassing Human-Level Performance in Person Re-Identification》理解
- Oracle创建用户和表空间
一.概述 1.数据库实际管理中,不同业务系统需要使用'不同的用户'进行管理维护和使用,这样做把业务数据和系统数据独立分开管理,利于数据库系统管理: 2.在数据库中创建业务系统用户时候,建议为用户创建指 ...
- 最优化:凸集、凸函数、KKT条件极其解释
1.凸集(大概定义) 2.凸函数 3.KK条件