【LOJ#575】【LNR#2】不等关系(容斥,动态规划,分治FFT)

时间:2022-09-21 23:05:15

【LOJ#575】【LNR#2】不等关系(容斥,动态规划,分治FFT)

题面

LOJ

题解

一个暴力\(dp\),设\(f[i][j]\)表示考虑完了前\(i\)个位置,其中最后一个数在前面所有数中排名是第\(j\)大,那么转移的时候枚举一下当前数是第几大,并且满足不等式的限制就可以了,然后拿前缀和优化一下就可以做到\(O(n^2)\)。

我们把所有连续的<看成一段,这样子题目就变成了每次要选出一段连续的上升序列,然后相邻两个连续段之间必须满足前一段的末尾要大于后一段的开头。

显然这个大于号是不好处理的,如果我们能够任意就很好做了。

这些>,即段与段之间的分割的位置的大小情况,如果至少有\(i\)个随意,方案数是\(g[i]\),那么对于答案的贡献就是\((-1)^ig[i]\)。

再考虑一个\(dp\),我们假设分割出来的所有段中,第\(i\)段的长度是\(a[i]\)。设\(f[i][j]\)表示考虑完了前\(i\)段,选出了\(j\)个上升序列的方案数,这样子就至少有\(i-j\)个位置是非法的。转移的话枚举把哪一段作为一段上升序列,那么就是:

\[f[i][j]=\sum_{k=0}^{i-1}f[k][j-1]*{n-s[k]\choose s[i]-s[k]}
\]

其中\(s\)是\(a\)的前缀和。

不难发现第二维用处不大,因为容斥系数只有\(\pm 1\),所以可以把容斥系数带进去直接带进去而不需要额外记录第二维来辅助计算。

于是转移就变成了:

\[f[i]=-\sum_{k=0}^{i-1}f[k]{n-s[k]\choose s[i]-s[k]}
\]

发现拆开之后可以卷积,然后有一项是\((s[i]-s[k])!\)不太好弄,因为\(s\)足够小,所以把\(i\)变到\(s[i]\)位置卷,这样子\(s[i]-s[k]\)就变成了\(i-k\),那么对于非\(s[i]\)的位置,把它强制弄成\(0\)。这样子拿分治\(FFT\)卷一下就好了。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MOD 998244353
#define MAX 524288
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
int fpow(int a,int b){int s=1;while(b){if(b&1)s=1ll*s*a%MOD;a=1ll*a*a%MOD;b>>=1;}return s;}
int W[MAX],r[MAX];
void NTT(int *P,int opt,int len)
{
int N,l=0;for(N=1;N<len;N<<=1)++l;
for(int i=0;i<N;++i)r[i]=(r[i>>1]>>1)|((i&1)<<(l-1));
for(int i=0;i<N;++i)if(i<r[i])swap(P[i],P[r[i]]);
for(int i=1;i<N;i<<=1)
{
int w=fpow(3,(MOD-1)/(i<<1));W[0]=1;
for(int k=1;k<i;++k)W[k]=1ll*W[k-1]*w%MOD;
for(int j=0,p=i<<1;j<N;j+=p)
for(int k=0;k<i;++k)
{
int X=P[j+k],Y=1ll*W[k]*P[i+j+k]%MOD;
P[j+k]=(X+Y)%MOD;P[i+j+k]=(X+MOD-Y)%MOD;
}
}
if(opt==-1)
{
reverse(&P[1],&P[N]);
for(int i=0,inv=fpow(N,MOD-2);i<N;++i)P[i]=1ll*P[i]*inv%MOD;
}
}
int jc[MAX],jv[MAX],inv[MAX];
int n,a[MAX],ans,cnt;char s[MAX];
bool book[MAX];
int A[MAX],B[MAX],f[MAX];
void CDQ(int l,int r)
{
if(l==r)
{
if(l==0)f[l]=1;
else if(!book[l])f[l]=0;
else f[l]=1ll*f[l]*(MOD-jv[n-l])%MOD;
return;
}
int mid=(l+r)>>1;
CDQ(l,mid);
for(int i=l;i<=mid;++i)A[i-l]=1ll*f[i]*jc[n-i]%MOD;
for(int i=1;i<=r-l+1;++i)B[i]=jv[i];
int N;for(N=1;N<=r-l+1+mid-l+1;N<<=1);
NTT(A,1,N);NTT(B,1,N);
for(int i=0;i<N;++i)A[i]=1ll*A[i]*B[i]%MOD;
NTT(A,-1,N);
for(int i=mid+1;i<=r;++i)f[i]=(f[i]+A[i-l])%MOD;
for(int i=0;i<N;++i)A[i]=B[i]=0;
CDQ(mid+1,r);
}
int C(int n,int m){if(n<0||m<0||n<m)return 0;return 1ll*jc[n]*jv[m]%MOD*jv[n-m]%MOD;}
int main()
{
scanf("%s",s+1);n=strlen(s+1)+1;
for(int i=1;i<n;++i)if(s[i]=='>')book[i]=true,++cnt;
inv[0]=inv[1]=jc[0]=jv[0]=1;
for(int i=2;i<=n;++i)inv[i]=1ll*inv[MOD%i]*(MOD-MOD/i)%MOD;
for(int i=1;i<=n;++i)jc[i]=1ll*jc[i-1]*i%MOD;
for(int i=1;i<=n;++i)jv[i]=1ll*jv[i-1]*inv[i]%MOD;
CDQ(0,n);
/*
f[0]=1;
for(int i=1;i<=n;++i)
if(book[i])
for(int j=0;j<i;++j)
f[i]=(f[i]+MOD-1ll*f[j]*C(n-j,i-j)%MOD)%MOD;
*/
for(int i=0;i<=n;++i)ans=(ans+f[i])%MOD;
if(cnt&1)ans=(MOD-ans)%MOD;
printf("%d\n",ans);
return 0;
}

【LOJ#575】【LNR#2】不等关系(容斥,动态规划,分治FFT)的更多相关文章

  1. LOJ575&period; 「LibreOJ NOI Round &num;2」不等关系 &lbrack;容斥,分治FFT&rsqb;

    LOJ 思路 发现既有大于又有小于比较难办,使用容斥,把大于改成任意减去小于的. 于是最后的串就长成这样:<<?<?<??<<<?<.我们把一段连续的& ...

  2. 【洛谷5644】&lbrack;PKUWC2018&rsqb; 猎人杀(容斥&plus;生成函数&plus;分治NTT)

    点此看题面 大致题意: 有\(n\)个人相互开枪,每个人有一个仇恨度\(a_i\),每个人死后会开枪再打死另一个还活着的人,且第一枪由你打响.设当前剩余人仇恨度总和为\(k\),则每个人被打中的概率为 ...

  3. 消失之物(背包DP)(容斥或分治)

    容斥做法: 首先n^2搞出f[i][j]第i个物品,j体积的方案数. 去除每个物品贡献: 设个g[i][j]表示当i不选,j体积方案数(注意不是此时的范围相对于全局,而不是1---i) 那么我们用到一 ...

  4. &lbrack;LOJ&num;3119&rsqb;&lbrack;Luogu5400&rsqb;&lbrack;CTS2019&rsqb;随机立方体&lpar;容斥&plus;DP&rpar;

    https://www.cnblogs.com/cjyyb/p/10900993.html #include<cstdio> #include<algorithm> #defi ...

  5. 【LOJ&num;2542】&lbrack;PKUWC2018&rsqb;随机游走(min-max容斥,动态规划)

    [LOJ#2542][PKUWC2018]随机游走(min-max容斥,动态规划) 题面 LOJ 题解 很明显,要求的东西可以很容易的进行\(min-max\)容斥,那么转为求集合的\(min\). ...

  6. 【LOJ&num;6072】苹果树(矩阵树定理,折半搜索,容斥)

    [LOJ#6072]苹果树(矩阵树定理,折半搜索,容斥) 题面 LOJ 题解 emmmm,这题似乎猫讲过一次... 显然先\(meet-in-the-middle\)搜索一下对于每个有用的苹果数量,满 ...

  7. LOJ &num;2541&period; 「PKUWC 2018」猎人杀&lpar;容斥 &comma; 期望dp &comma; NTT优化&rpar;

    题意 LOJ #2541. 「PKUWC 2018」猎人杀 题解 一道及其巧妙的题 , 参考了一下这位大佬的博客 ... 令 \(\displaystyle A = \sum_{i=1}^{n} w_ ...

  8. 【LOJ&num;6374】网格(二项式反演,容斥)

    [LOJ#6374]网格(二项式反演,容斥) 题面 LOJ 要从\((0,0)\)走到\((T_x,T_y)\),每次走的都是一个向量\((x,y)\),要求\(0\le x\le M_x,0\le ...

  9. LOJ &num;2542 &lbrack;PKUWC2018&rsqb;随机游走 &lpar;概率期望、组合数学、子集和变换、Min-Max容斥&rpar;

    很好很有趣很神仙的题! 题目链接: https://loj.ac/problem/2542 题意: 请自行阅读 题解首先我们显然要求的是几个随机变量的最大值的期望(不是期望的最大值),然后这玩意很难求 ...

随机推荐

  1. 设置easyui input默认值

    /*设置input 焦点*/ $(function () { //集体调用 $(".formTextBoxes input").each(function () { $(this) ...

  2. PHP json不转义

    json_encode($result, JSON_UNESCAPED_UNICODE);

  3. thinkPHP中&lowbar;initialize方法实例分析

    子类的_initialize方法自动调用父类的_initialize方法. 而php的构造函数construct,如果要调用父类的方法,必须在子类构造函数显示调用parent::__construct ...

  4. Windows Subsystem for Linux 环境变量

    WSL(Windows Subsystem for Linux )的环境变量是包含Linux子系统和Windows系统的,测试如下: wy@WY-PC:/mnt/c/Windows/System32$ ...

  5. 海康威视笔试(C&plus;&plus;)

    1. select和epoll的区别 2.服务器并发量之高性能服务器设计 3.SQL关键字 4.TCP乱序和重传的问题 5.c++对象内存分配问题 6.c++多线程 join的用法: Thread类的 ...

  6. codeforces231C

    To Add or Not to Add CodeForces - 231C A piece of paper contains an array of n integers a1, a2, ..., ...

  7. 装机uefi问题

    1 uefi不支持普通winpe启动,需要改为legacy 2 七代酷睿不支持win10以下版本 3 装机后无法启动是因为隐藏分区问题,删除掉,最好整个硬盘格式化 4 老毛桃一键装机不可以时,改为虚拟 ...

  8. WebForms UnobtrusiveValidationMode requires a ScriptResourceMapping for &&num;39&semi;jquery&&num;39&semi;&period; Please add a ScriptResourceMapping named jquery&lpar;case-sensitive&rpar;&period;

    新开一个Web site.没有使用jQuery,当Insus.NET使用一些验证控件时,如RequiredfieldValidator,程序出现下面错误: WebForms UnobtrusiveVa ...

  9. s4-3 CSMA

    载波侦听多路访问协议  CSMA:Carrier Sense Multiple Access 特点:"先听后发" 改进ALOHA协议的侦听/发送策略  分类 非持续式 持 ...

  10. Python3高级用法综合举例

    [本文出自天外归云的博客园] 举例 下面代码围绕一个Student类综合举例说明装饰器.生成器.动态获取/添加类成员.列表推导式.reduce函数.lambda表达式的实际应用: from funct ...