正题

题目链接:https://www.luogu.com.cn/problem/CF848E


题目大意

\(2n\)个花排成一个圆环,\(n\)种颜色每种两个,要求两个相同颜色之间最小距离为\(1,2\)或\(n\)。

对于一种染色方案的权值为:删除掉距离为\(n\)的颜色后,剩下的连续段长度的乘积。

求所有方案的染色之和对\(998244353\)取模。

\(1\leq n\leq 50000\)


解题思路

环好像很麻烦,先考虑线段上的,现在有两个长度为\(n\)的数列,然后距离为\(n\)的点之间对应。染色可以看为连接两个点。

然后设\(g_i\)表示不使用跨越数列的连线,涂\(i\)个的方案数,那么有\(g_i=g_{i-2}+g_{i-4}\)(相邻的连接/两个都是隔着对方连)。

然后考虑有跨越数列的线的方案,且没有其他连线跨过这条线,\(f0_i\)表示第\(i\)个是满足条件的线的权值和。\(f1_i\)则表示刚好有一对距离为\(2\)的点对跨越这个线的权值和。

那么有转移方程

\[f0_i=g_ii^2+\sum_{j=0}^{i-1}g_jj^2f0_{i-j-1}+\sum_{j=0}^{i-3}g_j(j+1)^2f1_{i-j-3}
\]

(第一个是全程没有其他横跨边,第二个是上一条横跨边两边没有同色,第三个是上一条横跨边两边有同色)

同理可以得到\(f1\)的方程

\[f1_i=g_i(i+1)^2+\sum_{j=0}^{i-1}g_j(j+1)^2f0_{i-j-1}+\sum_{j=0}^{i-3}g_j(j+2)^2f1_{i-j-3}
\]

得到\(f0\)和\(f1\)之后,看一下\(f0,f1\)都是最左边没有距离为\(2\)的边越过的,但是我们转换到环上的时候需要考虑这种情况,所以我们设\(f2_i\)表示左右两边的横跨边都有同色的,中间距离为\(i\)的权值和。

方程是

\[f2_i=g_i(i+2)^2+\sum_{j=0}^{i-1}g_j(j+1)^2f0_{i-j-1}+\sum_{j=0}^{j-3}g_j(j+2)^2f1_{i-j-3}
\]

然后考虑转换到行上。

如果只有一个点对距离是\(n\),那么贡献是\((n-1)\),有\(n\)种旋转方法,如果这个点对两边没有同色点,那么方案数是\(g_{n-1}\),否则是\(g_{n-3}\),所以这种情况的方案是\((n-1)^2n(g_{n-1}+g_{n-3})\)

然后剩下的我们可以先固定\(1\sim n+1\),然后枚举第二个距离为\(n\)的点对。设为\(i\),那贡献就是$$i(i-1)^2(g_{i-1}f0_{n-i-1}+2g_{i-2}f1_{n-i-2}+g_{i-3}f2_{n-i-3})$$

然后前面求\(f0,f1,f2\)都可以用分治\(NTT\)搞。

时间复杂度\(O(n\log^2 n)\)

如果用生成函数再推推可以分别得到\(O(n)\)和\(O(n\log n)\)的方法。

还有可以用生成函数发现这是一个\(16\)项的线性递推式,打出前面的表再用高斯消元得到系数,可以把时间优化到\(log\)级别

路还很长啊


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const ll N=2e5+10,P=998244353;
ll n,m,r[N],g[N],h[3][N],f[3][N];
ll t[3][N],T[2][N],z[4][N];
ll power(ll x,ll b){
ll ans=1;
while(b){
if(b&1)ans=ans*x%P;
x=x*x%P;b>>=1;
}
return ans;
}
void Glen(ll n){
m=1;while(m<=n)m<<=1;
for(ll i=0;i<m;i++)
r[i]=(r[i>>1]>>1)|((i&1)?(m>>1):0);
return;
}
void NTT(ll *f,ll op){
for(ll i=0;i<m;i++)
if(i<r[i])swap(f[i],f[r[i]]);
for(ll p=2;p<=m;p<<=1){
ll len=(p>>1),tmp=power(3,(P-1)/p);
if(op==-1)tmp=power(tmp,P-2);
for(ll k=0;k<m;k+=p){
ll buf=1;
for(ll i=k;i<k+len;i++){
ll tt=buf*f[i+len]%P;
f[i+len]=(f[i]-tt+P)%P;
f[i]=(f[i]+tt)%P;
buf=buf*tmp%P;
}
}
}
if(op==-1){
ll inv=power(m,P-2);
for(ll i=0;i<m;i++)
f[i]=f[i]*inv%P;
}
return;
}
void CDQ(ll l,ll r){
if(l==r){
(f[0][l]+=h[0][l])%=P;
(f[1][l]+=h[1][l])%=P;
return;
}
ll mid=(l+r)>>1;CDQ(l,mid);
Glen((r-l+1)*2);
for(ll i=0;i<m;i++)
t[0][i]=t[1][i]=t[2][i]=T[0][i]=T[1][i]=0;
for(ll i=0;i<=r-l+1;i++)
t[0][i]=h[0][i],t[1][i]=h[1][i],t[2][i]=h[2][i];
for(ll i=0;i<=mid-l;i++)
T[0][i]=f[0][i+l],T[1][i]=f[1][i+l];
NTT(t[0],1);NTT(t[1],1);NTT(t[2],1);
NTT(T[0],1);NTT(T[1],1);
for(ll i=0;i<m;i++){
z[0][i]=t[0][i]*T[0][i]%P,z[1][i]=t[1][i]*T[1][i]%P;
z[2][i]=t[1][i]*T[0][i]%P,z[3][i]=t[2][i]*T[1][i]%P;
}
NTT(z[0],-1);NTT(z[1],-1);NTT(z[2],-1);NTT(z[3],-1);
for(ll i=0;i<=r-l+1;i++){
if(l+i+1>mid&&l+i+1<=r){
f[0][l+i+1]=(f[0][l+i+1]+z[0][i])%P;
f[1][l+i+1]=(f[1][l+i+1]+z[2][i])%P;
}
if(l+i+3>mid&&l+i+3<=r){
f[0][l+i+3]=(f[0][l+i+3]+z[1][i])%P;
f[1][l+i+3]=(f[1][l+i+3]+z[3][i])%P;
}
}
CDQ(mid+1,r);return;
}
void solve(ll l,ll r){
if(l==r){(f[2][l]+=h[2][l])%=P;return;}
ll mid=(l+r)>>1;solve(l,mid);
Glen((r-l+1)*2);
for(ll i=0;i<m;i++)
t[1][i]=t[2][i]=T[0][i]=T[1][i]=0;
for(ll i=0;i<=r-l+1;i++)
t[1][i]=h[1][i],t[2][i]=h[2][i];
for(ll i=0;i<=mid-l;i++)
T[0][i]=f[1][l+i],T[1][i]=f[2][l+i];
NTT(t[1],1);NTT(t[2],1);NTT(T[0],1);NTT(T[1],1);
for(ll i=0;i<m;i++){
z[0][i]=t[1][i]*T[0][i]%P;
z[1][i]=t[2][i]*T[1][i]%P;
}
NTT(z[0],-1);NTT(z[1],-1);
for(ll i=0;i<r-l+1;i++){
if(l+i+1>mid&&l+i+1<=r)
(f[2][l+i+1]+=z[0][i])%=P;
if(l+i+3>mid&&l+i+3<=r)
(f[2][l+i+3]+=z[1][i])%=P;
}
solve(mid+1,r);
return;
}
signed main()
{
freopen("a.in","r",stdin);
freopen("a.out","w",stdout);
scanf("%lld",&n);g[0]=g[2]=1;
for(ll i=4;i<=n;i++)
g[i]=(g[i-4]+g[i-2])%P;
for(ll i=0;i<=n;i++){
h[0][i]=g[i]*i%P*i%P;
h[1][i]=g[i]*(i+1)%P*(i+1)%P;
h[2][i]=g[i]*(i+2)%P*(i+2)%P;
}
CDQ(0,n);
solve(0,n);
ll ans=(g[n-1]+g[n-3])*(n-1)%P*(n-1)%P*n%P;
for(ll i=2;i<n-1;i++){
ll tmp=g[i-1]*f[0][n-i-1]%P;
tmp=(tmp+2*g[i-2]*f[1][n-i-2]%P)%P;
tmp=(tmp+g[i-3]*f[2][n-i-3]%P)%P;
tmp=tmp*i%P*(i-1)%P*(i-1)%P;
ans=(ans+tmp)%P;
}
printf("%lld\n",ans);
return 0;
}

CF848E-Days of Floral Colours【dp,分治NTT】的更多相关文章

  1. CF848E Days of Floral Colours——DP+多项式求逆/分治NTT

    官方题解:http://codeforces.com/blog/entry/54233 就是由简入繁 1.序列处理,只考虑一个半圆 2.环形处理(其实这个就是多了旋转同构) 然后基于分割线邻居的跨越与 ...

  2. Codeforces 848E - Days of Floral Colours(分治 FFT)

    Codeforces 题目传送门 & 洛谷题目传送门 神仙 D1E,一道货真价实的 *3400 %%%%%%%%%%%% 首先注意到一点,由于该图为中心对称图形,\(1\sim n\) 的染色 ...

  3. HDU 5279 YJC plays Minecraft (分治NTT优化DP)

    题目传送门 题目大意:有$n$个小岛,每个小岛上有$a_{i}$个城市,同一个小岛上的城市互相连接形成一个完全图,第$i$个小岛的第$a_{i}$个城市和第$i+1$个小岛的第$1$个城市连接,特别地 ...

  4. HDU 5322 Hope (分治NTT优化DP)

    题面传送门 题目大意: 假设现在有一个排列,每个数和在它右面第一个比它大的数连一条无向边,会形成很多联通块. 定义一个联通块的权值为:联通块内元素数量的平方. 定义一个排列的权值为:每个联通块的权值之 ...

  5. ZOJ 3874 Permutation Graph (分治NTT优化DP)

    题面:vjudge传送门 ZOJ传送门 题目大意:给你一个排列,如果两个数构成了逆序对,就在他们之间连一条无向边,这样很多数会构成一个联通块.现在给出联通块内点的编号,求所有可能的排列数 推来推去容易 ...

  6. 【BZOJ-3456】城市规划 CDQ分治 + NTT

    题目链接 http://www.lydsy.com/JudgeOnline/problem.php?id=3456 Solution 这个问题可以考虑dp,利用补集思想 N个点的简单图总数量为$2^{ ...

  7. CF960G Bandit Blues 分治+NTT(第一类斯特林数)

    $ \color{#0066ff}{ 题目描述 }$ 给你三个正整数 \(n\),\(a\),\(b\),定义 \(A\) 为一个排列中是前缀最大值的数的个数,定义 \(B\) 为一个排列中是后缀最大 ...

  8. CF 848E(动态规划+分治NTT)

    传送门: http://codeforces.com/problemset/problem/848/E 题解: 假设0-n一定有一条边,我们得到了一个方案,那么显然是可以旋转得到其他方案的. 记最大的 ...

  9. Codeforces 1553I - Stairs(分治 NTT+容斥)

    Codeforces 题面传送门 & 洛谷题面传送门 u1s1 感觉这道题放到 D1+D2 里作为 5250 分的 I 有点偏简单了吧 首先一件非常显然的事情是,如果我们已知了排列对应的阶梯序 ...

随机推荐

  1. .Net Core 集成 Redis

    首先安装RedisServer 安装教程可参照 http://www.redis.cn/download.html 或者 https://www.runoob.com/redis/redis-inst ...

  2. dataTemplate 使用

    App ----------------------------------------------------------------- <Application x:Class=" ...

  3. SSM整合二

    总结 <!-- 批量删除 --> <delete id="deleteAll"> delete from tbl_emp where emp_id in & ...

  4. (三)ES6基础语法。。。freecodecamp笔记

    ECMAScript6 ECMAScript 是 JavaScript 的标准化版本,它旨在统一语言的规范和功能.所有主流的浏览器或者 Javascript 的运行环境都支持这个规范,因此 ECMAS ...

  5. Struts2之文件上传与下载

    时间:2017-1-11 15:47 --使用commons-fileupload组件上传1.客户端    *   method="post"    *   <input t ...

  6. Git(GitHub)配合TortoiseGit使用

    1.首先下载安装配置Git 安装请参照 https://www.cnblogs.com/xueweisuoyong/p/11914045.html 配置请参照 https://www.jianshu. ...

  7. rollup 使用babel7版本的插件rollup-plugin-babel,rollup-plugin-babel使用报错解决办法。

    最近在研究rollup,想吐槽下rollup的官方文档写的真的太简单了,而且照着文档一步步来还报错,说明文档年代有点久远啊... 照着文档使用rollup-plugin-babel报错,首先打开rol ...

  8. 为开源项目 go-gin-api 增加后台任务模块

    目录 任务管理界面 (WEB) 任务调度器 任务执行器 小结 推荐阅读 任务管理界面 (WEB) 支持在 WEB 界面 中对任务进行管理,例如:新增任务.编辑任务.启用/禁用任务.手动执行任务 等. ...

  9. NOIP模拟21:「Median·Game·Park」

    T1:Median   线性筛+桶+随机化(??什么鬼?).   首先,题解一句话秀到了我: 考虑输入如此诡异,其实可以看作随机数据   随机数据??   这就意味着分布均匀..   又考虑到w< ...

  10. 2021.9.12周六PAT甲级考试复盘与总结

    周六PAT甲级考试复盘与总结 先说结论:仍未步入"高手"行列:现在的学习节奏与方法是对的,有十万分的必要坚持下去. 题目 知识点 分数 T1 前缀和.二分 11 / 20 T2 排 ...