题目链接

正解:子集和变换。

考场上只会暴力和$p=0$的情况,还只会$O(2^{n}*n^{3})$的。

然而这题题面出锅,导致考场上一直在卡裸暴力,后面的部分分没写了。。听$laofu$说$O(2^{n}*n^{3})$可以过。。

所以直接讲正解。。

我们假设每个城市可以在两个不同集合,那么可以把子集卷积变成或卷积。

我们只要记下当前总共有多少个点,于是考虑设$f[i][S]$表示$i$个点,集合为$S$的方案数。

最后的$f[n][all]$就是答案,显然这个状态中的每个城市只会出现一次。

那么$f[i][S]=\sum f[j][T]*(\frac{sum[A]}{sum[S]})^{p}$,其中$A|T=S$,且$A$是一个合法集合。

可以把分母移项到左边,然后我们可以设$g[i][S]$表示如果$S$是一个合法集合,且$S$的位数为$i$,那么$g[i][S]=sum[S]^{p}$,否则为$0$。

那么$f[i][S]*sum[S]^{p}=\sum f[j][T]*g[i-j][A]$。注意到这个式子可以直接$FMT$以后点乘,再$IFMT$回来以后作除法得到,总复杂度为$O(2^{n}*n^{2})$。

感觉这道题其实并没有那么难,但是考场上被题面以及固定的套路给局限住了,所以并没有想到可以交换$dp$的两维状态从而优化复杂度。

 #include <bits/stdc++.h>
#define il inline
#define RG register
#define ll long long
#define rhl (998244353)
#define N (1<<21|1) using namespace std; int fa[],g[],w[N],cnt[N],can[N],inv[],n,m,p,all;
int f[][N],h[][N]; il int gi(){
RG int x=,q=; RG char ch=getchar();
while ((ch<'' || ch>'') && ch!='-') ch=getchar();
if (ch=='-') q=-,ch=getchar();
while (ch>='' && ch<='') x=x*+ch-'',ch=getchar();
return q*x;
} il int qpow(RG int a,RG int b){
if (!b) return ;
if (b==) return a;
return 1LL*a*a%rhl;
} il void fmt(int *a){
for (RG int i=;i<all;i<<=)
for (RG int j=;j<all;++j){
if (j&i) a[j]+=a[j^i];
if (a[j]>=rhl) a[j]-=rhl;
}
return;
} il void ifmt(int *a){
for (RG int i=;i<all;i<<=)
for (RG int j=;j<all;++j){
if (j&i) a[j]-=a[j^i];
if (a[j]<) a[j]+=rhl;
}
return;
} il int find(RG int x){
return fa[x]==x ? x : fa[x]=find(fa[x]);
} int main(){
#ifndef ONLINE_JUDGE
freopen("walk.in","r",stdin);
freopen("walk.out","w",stdout);
#endif
n=gi(),m=gi(),p=gi(),all=<<n,inv[]=;
for (RG int i=,u,v;i<=m;++i)
u=gi()-,v=gi()-,g[u]|=<<v,g[v]|=<<u;
for (RG int i=;i<=;++i)
inv[i]=1LL*(rhl-rhl/i)*inv[rhl%i]%rhl;
for (RG int i=;i<n;++i) w[<<i]=gi(); fmt(w);
for (RG int i=;i<all;++i) cnt[i]=cnt[i>>]+(i&);
for (RG int i=,fg,lst;i<all;++i){
fg=,lst=-;
for (RG int j=;j<n;++j) fa[j]=j;
for (RG int j=;j<n;++j){
if (!(i>>j&)) continue;
for (RG int k=,x,y;k<n;++k){
if (!(i>>k&) || !(g[j]>>k&)) continue;
x=find(j),y=find(k); if (x!=y) fa[x]=y;
}
}
for (RG int j=;j<n;++j){
if (i>>j&){
if (lst==-) lst=find(j);
else if (lst!=find(j)){ fg=; break; }
}
if ((i>>j&) && (cnt[g[j]&i]&)){ fg=; break; }
}
can[i]=fg;
}
for (RG int i=;i<all;++i) if (can[i]) h[cnt[i]][i]=qpow(w[i],p);
for (RG int i=;i<=n;++i) fmt(h[i]); f[][]=,fmt(f[]);
for (RG int i=;i<=n;++i){
int *F=f[i];
for (RG int j=;j<i;++j){
int *a=f[j],*b=h[i-j];
for (RG int s=;s<all;++s)
F[s]=(1LL*a[s]*b[s]+F[s])%rhl;
}
ifmt(F);
for (RG int s=;s<all;++s)
F[s]=i==cnt[s]?1LL*F[s]*qpow(inv[w[s]],p)%rhl:;
if (i^n) fmt(F);
}
cout<<f[n][all-]; return ;
}

bzoj5153 [Wc2018]州区划分的更多相关文章

  1. [WC2018]州区划分——FWT+DP+FST

    题目链接: [WC2018]州区划分 题目大意:给n个点的一个无向图,点有点权,要求将这n个点划分成若干个部分,每部分合法当且仅当这部分中所有点之间的边不能构成欧拉回路.对于一种划分方案,第i个部分的 ...

  2. [WC2018]州区划分

    [WC2018]州区划分 注意审题: 1.有序选择 2.若干个州 3.贡献是州满意度的乘积 枚举最后一个州是哪一个,合法时候贡献sum[s]^p,否则贡献0 存在欧拉回路:每个点都是偶度数,且图连通( ...

  3. [UOJ#348][WC2018]州区划分

    [UOJ#348][WC2018]州区划分 试题描述 小 \(S\) 现在拥有 \(n\) 座城市,第ii座城市的人口为 \(w_i\),城市与城市之间可能有双向道路相连. 现在小 \(S\) 要将这 ...

  4. [WC2018]州区划分(FWT,FST)

    [WC2018]州区划分(FWT,FST) Luogu loj 题解时间 经典FST. 在此之前似乎用到FST的题并不多? 首先预处理一个子集是不是欧拉回路很简单,判断是否连通且度数均为偶数即可. 考 ...

  5. P4221 [WC2018]州区划分 无向图欧拉回路 FST FWT

    LINK:州区划分 把题目中四个条件进行规约 容易想到不合法当前仅当当前状态是一个无向图欧拉回路. 充要条件有两个 联通 每个点度数为偶数. 预处理出所有状态. 然后设\(f_i\)表示组成情况为i的 ...

  6. [WC2018]州区划分(FWT)

    题目描述 题解 这道题的思路感觉很妙. 题目中有一个很奇怪的不合法条件,貌似和后面做题没有什么关系,所以我们先得搞掉它. 也就是判断一个点集是否合法,也就是判断这个点集是否存在欧拉回路. 如果存在欧拉 ...

  7. Luogu4221 WC2018州区划分(状压dp+FWT)

    合法条件为所有划分出的子图均不存在欧拉回路或不连通,也即至少存在一个度数为奇数的点或不连通.显然可以对每个点集预处理是否合法,然后就不用管这个奇怪的条件了. 考虑状压dp.设f[S]为S集合所有划分方 ...

  8. LOJ2340 [WC2018] 州区划分 【FMT】【欧拉回路】

    题目分析: 这题是WC的题??? 令 $g[S] = (\sum_{x \in S}w_x)^p$ $h[S] = g[S]$如果$S$不是欧拉回路 $d[S] = \frac{f[S]}{g[All ...

  9. [WC2018]州区划分(状压DP+FWT/FMT)

    很裸的子集反演模板题,套上一些莫名其妙的外衣. 先预处理每个集合是否合法,再作显然的状压DP.然后发现可以写成子集反演的形式,直接套模板即可. 子集反演可以看这里. 子集反演的过程就是多设一维代表集合 ...

随机推荐

  1. 微信小程序通过CODE换取session_key和openid

    微信小程序的用户信息获取需要请求微信的服务器,通过小程序提供的API在小程序端获取CODE,然后将CODE传入到我们自己的服务器,用我们的服务器来换取session_key和openid. 小程序端比 ...

  2. 撩课-Java每天10道面试题第4天

    撩课Java+系统架构 视频 点击开始学习 31.静态变量和实例变量的区别? 静态变量也叫类变量, 这种变量前加了static修饰符. 可以直接用类名调用, 也可以用对象调用, 而且所有对象的同一个类 ...

  3. js 数组常用的一些方法

    数组可以说是js经常会遇到的数据结构,以下我们对数组进行详细的学习! 一.数组的创建 var mycars = new Array(): || new Array(3);  || new Array( ...

  4. 认识Groovy

    什么是groovy: Groovy 是 JVM 的一个替代语言 —替代 是指可以用 Groovy 在 Java 平台上进行 Java 编程,使用方式基本与使用 Java 代码的方式相同. 在编写新应用 ...

  5. 项目中遇到的问题——jsp:include

    昨晚记错了,项目中用的是这个<jsp:attribute>,不过没关系,都差不多!原理是传参数 具体用法: 假设有两个tag文件  aaa 和 bbb aaa有两个属性:name  age ...

  6. 浏览器同源策略与ajax跨域方法汇总

    原文 什么是同源策略 如果你进行过前端开发,肯定或多或少会听说过.接触过所谓的同源策略.那么什么是同源策略呢? 要了解同源策略,首先得理解“源”.在这个语境下,源(origin)其实就是指的URL.所 ...

  7. hadoop fs -put localfile . 时出现如下错误: could only be replicated to 0 nodes, instead of 1

    hadoop fs -put localfile . 时出现如下错误:could only be replicated to 0 nodes, instead of 1网友的说法: 这个问题是由于没有 ...

  8. HEOI2017 游记

    你若安好,便是晴天. …… 人就像命运下的蝼蚁,谁也无法操控自己的人生. ——阮行止 …… Day 0 中午就要出发了,上午教练还搞了一场欢乐信心赛,然而还是挂惨了.T3是bzoj的原题,但是当时写的 ...

  9. 服务器学习笔记之servlet

    初衷 想学习下服务器这边的知识,制定了一条学习路线:java8--->servlet--->springMvc--->springBoot--->springCloud.在此当 ...

  10. js 随机打乱数组

    假如有一个数组: var arr1=['a','b','c','d','e','f','g']; 需要将它进行随机打乱,网上好多都是用: arr1.sort(function(){ return 0. ...