bzoj5153 [Wc2018]州区划分
正解:子集和变换。
考场上只会暴力和$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]州区划分的更多相关文章
- [WC2018]州区划分——FWT+DP+FST
题目链接: [WC2018]州区划分 题目大意:给n个点的一个无向图,点有点权,要求将这n个点划分成若干个部分,每部分合法当且仅当这部分中所有点之间的边不能构成欧拉回路.对于一种划分方案,第i个部分的 ...
- [WC2018]州区划分
[WC2018]州区划分 注意审题: 1.有序选择 2.若干个州 3.贡献是州满意度的乘积 枚举最后一个州是哪一个,合法时候贡献sum[s]^p,否则贡献0 存在欧拉回路:每个点都是偶度数,且图连通( ...
- [UOJ#348][WC2018]州区划分
[UOJ#348][WC2018]州区划分 试题描述 小 \(S\) 现在拥有 \(n\) 座城市,第ii座城市的人口为 \(w_i\),城市与城市之间可能有双向道路相连. 现在小 \(S\) 要将这 ...
- [WC2018]州区划分(FWT,FST)
[WC2018]州区划分(FWT,FST) Luogu loj 题解时间 经典FST. 在此之前似乎用到FST的题并不多? 首先预处理一个子集是不是欧拉回路很简单,判断是否连通且度数均为偶数即可. 考 ...
- P4221 [WC2018]州区划分 无向图欧拉回路 FST FWT
LINK:州区划分 把题目中四个条件进行规约 容易想到不合法当前仅当当前状态是一个无向图欧拉回路. 充要条件有两个 联通 每个点度数为偶数. 预处理出所有状态. 然后设\(f_i\)表示组成情况为i的 ...
- [WC2018]州区划分(FWT)
题目描述 题解 这道题的思路感觉很妙. 题目中有一个很奇怪的不合法条件,貌似和后面做题没有什么关系,所以我们先得搞掉它. 也就是判断一个点集是否合法,也就是判断这个点集是否存在欧拉回路. 如果存在欧拉 ...
- Luogu4221 WC2018州区划分(状压dp+FWT)
合法条件为所有划分出的子图均不存在欧拉回路或不连通,也即至少存在一个度数为奇数的点或不连通.显然可以对每个点集预处理是否合法,然后就不用管这个奇怪的条件了. 考虑状压dp.设f[S]为S集合所有划分方 ...
- 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 ...
- [WC2018]州区划分(状压DP+FWT/FMT)
很裸的子集反演模板题,套上一些莫名其妙的外衣. 先预处理每个集合是否合法,再作显然的状压DP.然后发现可以写成子集反演的形式,直接套模板即可. 子集反演可以看这里. 子集反演的过程就是多设一维代表集合 ...
随机推荐
- 利用jquery判断点是否在椭圆内
源码例子下载 : <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" ...
- C# 字符串处理—— 去除首位保留其他
//去除首位 public static string RemoveFirstPlace(string s) { ) //输入空值直接Return { ")) //判断开头是否是零 s = ...
- 2019-1-19 object祖宗类的equals重写
package com.test; /** * object祖宗类的equals重写 * @author Mr.kemi *2019-1-19 */ public class Equals { pri ...
- eclipse 更改背景颜色字体
原文 切一个自己的图: 废话不说,直接入题. 方式一:替换Eclipse的配置文件 其实Eclipse的各种配置都是在文件设置里的,因此只要用一个配置好的模版来替换默认的配置文件,即可将所有配置克隆到 ...
- js画一棵树
用纯js画一棵树.思路: 1.一棵树的图片,作为页面背景: 2.通过html5中的canvas画布进行遮罩: 3.定时每隔10ms,从下往上清除1px的遮罩: <!DOCTYPE html> ...
- MyBatis学习(三)---MyBatis和Spring整合
想要了解MyBatis基础的朋友可以通过传送门: MyBatis学习(一)---配置文件,Mapper接口和动态SQL http://www.cnblogs.com/ghq120/p/8322302. ...
- C++11:实用特性
今天逛cplusplus.com发现C++还真多了不少方便使用的特性,先了解些最常用的 初始化列表 vector<,,,}); vector<pair<int, int> &g ...
- qt5.6.3下使用firebird
有人把firebird比作数据库界的瑞士军刀,想学习一下其在QT5.6中的使用,于是便开始了一场自己挖坑,自己埋的旅程. 环境说明:win7 64位+QT5.6 mingw4.9 32位(好像官网上也 ...
- HTML5之新增的元素和废除的元素 (声明:内容节选自《HTML 5从入门到精通》)
新增结构元素: section元素 section元素定义文档或应用程序中的一个区段,比如章节.页眉.页脚或文档中的其他部分.它可以与h1,h2,h3,h4,h5,h6元素结合起来使用,标示文档结构. ...
- LintCode2016年8月8日算法比赛----子树
子树 题目描述 有两个不同大小的二叉树:T1有上百万的节点:T2有好几百的节点.请设计一种算法,判定T2是否为T1的子树. 注意事项 若 T1 中存在从节点 n 开始的子树与 T2 相同,我们称 T2 ...