ZOJ 3874 Permutation Graph (分治NTT优化DP)
题目大意:给你一个排列,如果两个数构成了逆序对,就在他们之间连一条无向边,这样很多数会构成一个联通块。现在给出联通块内点的编号,求所有可能的排列数
推来推去容易发现性质,同一联通块内的点一定是连续标号的,否则无解
然后我就不会了
好神的$NTT$优化$DP$啊
根据上面的性质,联通块之间是互不影响的,所以我们对每个联通块分别统计答案再相乘
定义$f[i]$表示$i$个点构成的合法联通块,可能的排列数
一个合法联通块的所有元素一定在同一联通块内,说明不可能存在两个联通块,因此
$f[i]=i!-\sum f[j]*(i-j)!$
发现这是一个卷积的形式,用分治$NTT$求解即可
模数是一个原根是10的费马素数
别忘了判断无解的情况
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N1 (1<<18)+10
#define il inline
#define dd double
#define ld long double
#define ll long long
using namespace std; const int inf=0x3f3f3f3f;
const ll p=;
int gint()
{
int ret=,fh=;char c=getchar();
while(c<''||c>''){if(c=='-')fh=-;c=getchar();}
while(c>=''&&c<=''){ret=ret*+c-'';c=getchar();}
return ret*fh;
}
ll qpow(ll x,ll y)
{
ll ans=;
for(;y;x=x*x%p,y>>=) if(y&) ans=ans*x%p;
return ans;
} namespace NTT{ ll a[N1],b[N1],c[N1],invwn[N1],mulwn[N1];
int r[][N1];
void Pre(int len,int L)
{
int i,j;
for(j=;j<=L;j++) for(i=;i<(<<j);i++)
r[j][i]=(r[j][i>>]>>)|((i&)<<(j-));
for(i=;i<=len;i<<=) mulwn[i]=qpow(,(p-)/i), invwn[i]=qpow(mulwn[i],p-);
}
void NTT(ll *s,int len,int type,int L)
{
int i,j,k; ll wn,w,t;
for(i=;i<len;i++) if(i<r[L][i]) swap(s[i],s[r[L][i]]);
for(k=;k<=len;k<<=)
{
wn=(type>)?mulwn[k]:invwn[k];
for(i=;i<len;i+=k)
{
for(j=,w=;j<(k>>);j++,w=w*wn%p)
{
t=w*s[i+j+(k>>)]%p;
s[i+j+(k>>)]=(s[i+j]+p-t)%p;
s[i+j]=(s[i+j]+t)%p;
}
}
}
}
void Main(int len,int L)
{
int i,invl=qpow(len,p-);
NTT(a,len,,L); NTT(b,len,,L);
for(i=;i<len;i++) c[i]=a[i]*b[i]%p;
NTT(c,len,-,L);
for(i=;i<len;i++) c[i]=c[i]*invl%p;
}
void clr(int sz)
{
memset(a,,sz<<);
memset(b,,sz<<);
} }; using NTT::a; using NTT::b; using NTT::c;
ll f[N1],g[N1]; int de;
void CDQ(int l,int r)
{
if(r-l<) return;
if(r-l==){ f[l]=(g[l]+p-f[l])%p; return; }
int mid=(l+r)>>,i,len,L;
CDQ(l,mid);
for(len=,L=;len<(mid-l)+(r-l)-;len<<=,L++);
for(i=l;i<mid;i++) NTT::a[i-l]=f[i];
for(i=;i<(r-l);i++) NTT::b[i]=g[i];
NTT::Main(len,L);
for(i=mid;i<r;i++) f[i]=(f[i]+NTT::c[i-l])%p;
NTT::clr(len);
CDQ(mid,r);
}
int T,n,m;
int que[N1]; int main()
{
int i,j,x,y,len,L,mi,ma; ll ans;
scanf("%d",&T); n=;
for(i=,g[]=;i<n;i++) g[i]=g[i-]*i%p;
for(len=,L=;len<n+n-;len<<=,L++);
NTT::Pre(len,L);
CDQ(,n); while(T--){ n=gint(); m=gint(); ans=;
for(i=;i<=m;i++)
{
x=gint(); mi=inf,ma=;
for(j=;j<=x;j++) que[j]=gint(), mi=min(mi,que[j]), ma=max(ma,que[j]);
if(ma-mi+!=x) ans=;
ans=(ans*f[x])%p;
}
printf("%lld\n",ans); }
return ; }
ZOJ 3874 Permutation Graph (分治NTT优化DP)的更多相关文章
- ZOJ 3874 Permutation Graph 分治NTT
Permutation Graph Time Limit: 2 Seconds Memory Limit: 65536 KB Edward has a permutation {a1, a2 ...
- ZOJ 3874 Permutation Graph ——分治 NTT
发现每一块一定是按照一定的顺序的. 然后与标号无关,并且相同大小的对答案的影响相同. 然后列出递推式,上NTT+分治就可以了. 然后就可以与输入同阶处理答案了. #include <map> ...
- HDU 5279 YJC plays Minecraft (分治NTT优化DP)
题目传送门 题目大意:有$n$个小岛,每个小岛上有$a_{i}$个城市,同一个小岛上的城市互相连接形成一个完全图,第$i$个小岛的第$a_{i}$个城市和第$i+1$个小岛的第$1$个城市连接,特别地 ...
- HDU 5322 Hope (分治NTT优化DP)
题面传送门 题目大意: 假设现在有一个排列,每个数和在它右面第一个比它大的数连一条无向边,会形成很多联通块. 定义一个联通块的权值为:联通块内元素数量的平方. 定义一个排列的权值为:每个联通块的权值之 ...
- 【uoj#244】[UER #7]短路 CDQ分治+斜率优化dp
题目描述 给出 $(2n+1)\times (2n+1)$ 个点,点 $(i,j)$ 的权值为 $a[max(|i-n-1|,|j-n-1|)]$ ,找一条从 $(1,1)$ 走到 $(2n+1,2n ...
- BZOJ1492:[NOI2007]货币兑换 (CDQ分治+斜率优化DP | splay动态维护凸包)
BZOJ1492:[NOI2007]货币兑换 题目传送门 [问题描述] 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和B纪念券(以下简称B券).每个持有金券的 ...
- 4.13 省选模拟赛 树 树形dp 卷积 NTT优化dp.
考试的时候 看到概率 看到期望我就怂 推了一波矩阵树推自闭了 发现 边权点权的什么也不是. 想到了树形dp 维护所有边的断开情况 然后发现数联通块的和再k次方过于困难. 这个时候 应该仔细观察一下 和 ...
- BZOJ 1492: [NOI2007]货币兑换Cash [CDQ分治 斜率优化DP]
传送门 题意:不想写... 扔链接就跑 好吧我回来了 首先发现每次兑换一定是全部兑换,因为你兑换说明有利可图,是为了后面的某一天两种卷的汇率差别明显而兑换 那么一定拿全利啊,一定比多天的组合好 $f[ ...
- bzoj3672/luogu2305 购票 (运用点分治思想的树上cdq分治+斜率优化dp)
我们都做过一道题(?)货币兑换,是用cdq分治来解决不单调的斜率优化 现在它放到了树上.. 总之先写下来dp方程,$f[i]=min\{f[j]+(dis[i]-dis[j])*p[i]+q[i]\} ...
随机推荐
- 关于bitnami redmine 的一些问题
http://blog.csdn.net/yanyaming920817/article/details/50059523
- 【ACM】poj_2080_Calendar_201307311043
CalendarTime Limit: 1000MS Memory Limit: 30000K Total Submissions: 9787 Accepted: 3677 Description ...
- 获取Class对象方式
在java中,每个class都有一个相应的Class对象,当编写好一个类,编译完成后,在生成的.class文件中,就产生一个Class对象,用来表示这个类的类型信息.获得Class实例的三种方式: 1 ...
- linux下apache2更换目录
修改apache2的默认文档目录(默认是在/var/www) 修改命令:sudo gedit /etc/apache2/sites-enabled/000-default 在文档中找到 Documen ...
- 【待解决】maven创建web报Cannot read lifecycle mapping metadata for artifact org.apache.maven.plugins
Cannot read lifecycle mapping metadata for artifact org.apache.maven.plugins:maven-war-plugin:maven- ...
- JavaScript高级特性之原型
JavaScript的原型 原型prototype属性仅仅适用于函数对象(这里的函数对象是自己为了理解更好定义的,普通对象是没有原型属性的) 1.研究函数原型: <script type=&qu ...
- luogu2261 [CQOI2007] 余数之和
题目大意 求 \[\sum_{i=1}^{n}(k\mod i)\] \(n,k\leq 10^9\). 题解 先只考虑\(n\leq k\)的情况. \[\sum_{i=1}^{n}(k\mod i ...
- vim下很好的右键复制方法
1)先按shift ,然后鼠标选中即可复制:(shift按下时为非vim环境) 2)好方法: "Enable and disable mouse use noremap <f1> ...
- 利用ajax,canvas实现的测试php程序占用内存的代码
receive.php <?php $array["time"]=time();$array["memory"]=memory_get_usage();e ...
- 转载【梦想天空(山边小溪)】Web 开发人员和设计师必读文章推荐【系列二十九】
博客地址:http://www.cnblogs.com/lhb25/p/must-read-links-for-web-designers-and-developers-volume-29.html