题面

有一个长为

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)/2​Ci−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 (组合数学)的更多相关文章

  1. 17995 Stupid thief 组合数学

    17995 Stupid thief 时间限制:1000MS  内存限制:65535K提交次数:0 通过次数:0 题型: 编程题   语言: 不限定 Description A stupid thie ...

  2. 记录在linux下的wine生活

    记录在linux下的windows生活 本篇内容涉及QQ.微信.Office的安装配置 QQ: 到deepin下载轻聊版. 如果安装了crossover,那么将其中opt/cxoffice/suppo ...

  3. linux Mint wine安装qq,桌面快捷键配置

    在桌面新建qq.desktop文件,添加以下内容 #!/usr/bin/env xdg-open[Desktop Entry]Comment=QQTerminal=falseName=QQExec=w ...

  4. Wine——在Linux上运行Windows软件

    官网:https://www.winehq.org/ 参考: wikipedia 教你使用Wine在Linux上运行Windows软件 如何安装和使用Wine,以便在Linux上运行Windows应用 ...

  5. 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 ...

  6. linux安装wine

    1.添加PPA sudo add-apt-repository ppa:ubuntu-wine/ppa 2.更新列表 sudo apt-get update 3.安装Wine sudo apt-get ...

  7. Codeforces632E Thief in a Shop(NTT + 快速幂)

    题目 Source http://codeforces.com/contest/632/problem/E Description A thief made his way to a shop. As ...

  8. CentOS 安装 Wine

    1. 下载安装包 Wine的中文官网可以下载到最新稳定和开发版本的Wine安装包,根据不同需求可以自行下载 2. 解压安装包,编译前检查 根据不同的平台选择不同的编译选项: For 32-Bit Sy ...

  9. wine

    sudo dpkg --add-architecture i386 sudo add-apt-repository ppa:wine/wine-buildssudo apt-get update su ...

随机推荐

  1. 【Redis】事件驱动框架源码分析(多线程)

    IO线程初始化 Redis在6.0版本中引入了多线程,提高IO请求处理效率. 在Redis Server启动函数main(server.c文件)中初始化服务之后,又调用了InitServerLast函 ...

  2. 你真的很了解printf函数吗?

    对C语言中经常使用的printf这个库函数,你是否真的吃透了呢? 系统化的学习C语言程序设计,是不是看过一两本C语言方面的经典著作就足够了呢?答案是显而易见的:不够.通过这种典型的入门级的学习方式,是 ...

  3. vue大型电商项目尚品汇(后台篇)day05

    今天继续是对后台管理部分的一个操作,但是快要结束了,今天结束,明天会进入一个从Vue以来,另外一个名声显著的东西了,一只耳闻从未见识,而且十分的炫酷 他就是------数据可视化Echarts,迫不及 ...

  4. Vue3.0系列——「vue3.0学习手册」第一期

    一.项目搭建 vite是尤大大开发的一款意图取代webpack的工具.其实现原理是利用ES6的import发送请求加载文件的特性.拦截这些请求,做一些编译,省去webpack冗长的打包时间.并将其与R ...

  5. MongoDB 的内存使用限制

    本文将简述一下MongoDB的内存限制问题 1. 使用Docker限制 当我们使用docker创建mongo 容器时,可通过使用以下参数,对mongo可以使用的资源进行限制 内存限制 参数 简介 -m ...

  6. VirtualBox虚拟机安装Ubuntu系统后,增加内存空间和处理器核心数

    对于Linux爱好者而言,初次使用虚拟机时,一般都会使用默认的设置,例如硬盘空间.内存空间等等. 而往往在熟悉之后,安装了某些必要的软件,或者熟悉了实际的开发场景后,却发现原本给虚拟机分配的物理资源是 ...

  7. Python实现12种概率分布(附代码)

    今天给大家带来的这篇文章是关于机器学习的,机器学习有其独特的数学基础,我们用微积分来处理变化无限小的函数,并计算它们的变化:我们使用线性代数来处理计算过程:我们还用概率论与统计学建模不确定性. 在这其 ...

  8. 《AlignedReID:Surpassing Human-Level Performance in Person Re-Identification》理解

  9. Oracle创建用户和表空间

    一.概述 1.数据库实际管理中,不同业务系统需要使用'不同的用户'进行管理维护和使用,这样做把业务数据和系统数据独立分开管理,利于数据库系统管理: 2.在数据库中创建业务系统用户时候,建议为用户创建指 ...

  10. 最优化:凸集、凸函数、KKT条件极其解释

    1.凸集(大概定义) 2.凸函数  3.KK条件