【UOJ348】【WC2018】州区划分 状压DP FWT
题目大意
给定一个\(n\)个点的无向图,对于每种 \(n\) 个点的划分\(\{S_1,S_2,\ldots,S_k\}\),定义它是合法的,当且仅当每个点都在其中的一个集合中且对于任何的\(i\in[1,k]\),点集\(S_i\)非空,且导出子图不存在欧拉回路。
给定数组\(w_i\),求对于所有合法的划分\(\{s_1,s_2,\ldots,s_k\}\),下面的式子之和
\]
\(n\leq 21\)
题解
先用\(O(n^22^n)\)判断每个点集是否合法,并计算\(g(S)={(\sum_{x\in S}w_x)}^p\)。如果\(S\)不合法。那么\(g(S)=0\)
很容易想到一个\(O(3^n)\)的做法。
设\(f(S)\)为当前选择的集合为\(S\)的答案
\]
可以发现这是一个子集卷积。
一种可行的做法是把子集卷积转化为子集或卷积。
定义\(\widetilde{f}\)是\(f\)的集合占位幂级数,当且仅当对于所有的\(S\)满足\(\widetilde{f}(S)\)是一个\(|S|\)次多项式,且\([x^{|S|}]\widetilde{f}(S)=f(S)\)
可以发现,若\(p(S)=\widetilde f(S)\cdot \widetilde g(S)\),则\(p\)是\(f\times g\)的占位多项式。
所以我们可以在\(O(n^22^n)\)内计算子集卷积了。
回到这道题,我们假设每个城市可以出现在多个州里,记\(h_{i,S}\)为每个州的城市个数之和为\(i\),每个州的城市的并集为\(S\)的方案数。那么\(F(S)=\sum_{i=1}^nh_{i,S}x^i\)就是\(f(S)\)的集合占位幂级数。
所以\(f(S)=h_{|S|,S}\),状态转移方程是
\]
记\(G(i)=\sum_{|S|=i}g(S)x^S\)
我们先枚举\(i\),再枚举\(j\),然后计算\(h_i=h_{i-j}\times G(j)\)
这里我们可以全程用莫比乌斯变换后的值,卷积一次就是\(O(2^n)\)的
还有,我们这里要除以\(g(S)\),可以在做完一层(\(h_i\))之后变换回去,除以\(g(S)\),再变换回来。
这样时间复杂度就是\(O(n^22^n)\)的了。
代码
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<ctime>
#include<cstdlib>
#include<utility>
using namespace std;
typedef long long ll;
typedef pair<int,int> pii;
void open(const char *s)
{
#ifndef ONLINE_JUDGE
char str[100];
sprintf(str,"%s.in",s);
freopen(str,"r",stdin);
sprintf(str,"%s.out",s);
freopen(str,"w",stdout);
#endif
}
const ll p=998244353;
ll fp(ll a,ll b)
{
ll s=1;
for(;b;b>>=1,a=a*a%p)
if(b&1)
s=s*a%p;
return s;
}
int lx[100010];
int ly[100010];
int n,m,o;
void fwt(int *a)
{
int i,j;
for(j=1;j<1<<n;j<<=1)
for(i=0;i<1<<n;i++)
if(i&j)
a[i]=(a[i]+a[i^j])%p;
}
void ifwt(int *a)
{
int i,j;
for(j=1;j<1<<n;j<<=1)
for(i=0;i<1<<n;i++)
if(i&j)
a[i]=(a[i]-a[i^j])%p;
}
int f[22][1<<21];
int g[22][1<<21];
ll inv[100010];
ll fw[1<<21];
ll fwi[1<<21];
int w[30];
int fa[100010];
int num[1<<21];
int find(int x)
{
return fa[x]==x?x:fa[x]=find(fa[x]);
}
int c[1<<21];
int d[100010];
int main()
{
open("walk");
int i,j,k;
inv[0]=inv[1]=1;
for(i=2;i<=10000;i++)
inv[i]=-p/i*inv[p%i]%p;
scanf("%d%d%d",&n,&m,&o);
for(i=1;i<=m;i++)
scanf("%d%d",&lx[i],&ly[i]);
for(i=1;i<=n;i++)
scanf("%d",&w[i]);
int all=(1<<n)-1;
for(i=1;i<=all;i++)
{
for(j=1;j<=n;j++)
{
d[j]=0;
fa[j]=j;
}
for(j=1;j<=m;j++)
if(((i>>(lx[j]-1))&1)&&((i>>(ly[j]-1))&1))
{
d[lx[j]]^=1;
d[ly[j]]^=1;
int fx=find(lx[j]);
int fy=find(ly[j]);
if(fx!=fy)
fa[fx]=fy;
}
fw[i]=0;
for(j=1;j<=n;j++)
if((i>>(j-1))&1)
fw[i]+=w[j];
fwi[i]=inv[fw[i]];
fw[i]=fp(fw[i],o);
fwi[i]=fp(fwi[i],o);
int cnt=0;
c[i]=0;
for(j=1;j<=n;j++)
if((i>>(j-1))&1)
{
num[i]++;
if(fa[j]==j)
{
cnt++;
if(cnt>=2)
c[i]=1;
}
if(d[j])
c[i]=1;
}
fw[i]*=c[i];
g[num[i]][i]=fw[i];
}
f[0][0]=1;
for(i=1;i<=n;i++)
fwt(g[i]);
fwt(f[0]);
for(i=1;i<=n;i++)
{
for(j=1;j<=i;j++)
for(k=0;k<=all;k++)
f[i][k]=(f[i][k]+(ll)f[i-j][k]*g[j][k])%p;
ifwt(f[i]);
for(j=0;j<=all;j++)
f[i][j]=f[i][j]*fwi[j]%p;
fwt(f[i]);
}
ifwt(f[n]);
ll ans=f[n][all];
ans=(ans+p)%p;
printf("%lld\n",ans);
return 0;
}
【UOJ348】【WC2018】州区划分 状压DP FWT的更多相关文章
- [WC2018]州区划分(状压DP+FWT/FMT)
很裸的子集反演模板题,套上一些莫名其妙的外衣. 先预处理每个集合是否合法,再作显然的状压DP.然后发现可以写成子集反演的形式,直接套模板即可. 子集反演可以看这里. 子集反演的过程就是多设一维代表集合 ...
- UOJ348 WC2018 州区划分 状压DP、欧拉回路、子集卷积
传送门 应该都会判欧拉回路吧(雾 考虑状压DP:设\(W_i\)表示集合\(i\)的点的权值和,\(route_i\)表示点集\(i\)的导出子图中是否存在欧拉回路,\(f_i\)表示前若干个城市包含 ...
- UOJ #348 州区划分 —— 状压DP+子集卷积
题目:http://uoj.ac/problem/348 一开始可以 3^n 子集DP,枚举一种状态的最后一个集合是什么来转移: 设 \( f[s] \) 表示 \( s \) 集合内的点都划分好了, ...
- Luogu4221 WC2018州区划分(状压dp+FWT)
合法条件为所有划分出的子图均不存在欧拉回路或不连通,也即至少存在一个度数为奇数的点或不连通.显然可以对每个点集预处理是否合法,然后就不用管这个奇怪的条件了. 考虑状压dp.设f[S]为S集合所有划分方 ...
- 集合划分状压dp
给一个 $n$ 个点 $m$ 条边的无向图,每条边有 $p_i$ 的概率消失,求图连通的概率 $n \leq 9$ sol: 我们考虑一个 $dp$ $f_{(i,S)}$ 表示只考虑前 $i$ 条边 ...
- P4221 [WC2018]州区划分 无向图欧拉回路 FST FWT
LINK:州区划分 把题目中四个条件进行规约 容易想到不合法当前仅当当前状态是一个无向图欧拉回路. 充要条件有两个 联通 每个点度数为偶数. 预处理出所有状态. 然后设\(f_i\)表示组成情况为i的 ...
- [WC2018]州区划分——FWT+DP+FST
题目链接: [WC2018]州区划分 题目大意:给n个点的一个无向图,点有点权,要求将这n个点划分成若干个部分,每部分合法当且仅当这部分中所有点之间的边不能构成欧拉回路.对于一种划分方案,第i个部分的 ...
- [WC2018]州区划分(FWT,FST)
[WC2018]州区划分(FWT,FST) Luogu loj 题解时间 经典FST. 在此之前似乎用到FST的题并不多? 首先预处理一个子集是不是欧拉回路很简单,判断是否连通且度数均为偶数即可. 考 ...
- [WC2018]州区划分(FWT)
题目描述 题解 这道题的思路感觉很妙. 题目中有一个很奇怪的不合法条件,貌似和后面做题没有什么关系,所以我们先得搞掉它. 也就是判断一个点集是否合法,也就是判断这个点集是否存在欧拉回路. 如果存在欧拉 ...
随机推荐
- duxing201606的原味鸡树
链接 [http://murphyc.fun/problem/4011] 题意 描述 众所周知,duxing哥非常喜欢原味鸡.众所周知,原味鸡是长在原味鸡树上的. duxing哥因为是水产巨子,所以就 ...
- HashSet中存放不重复元素
一.自定义对象存放在hashSet中,保证元素不重复.重写hashCode()和equals()方法 public class Student{ private String name; privat ...
- 四、Object.defineProperty总结
Object.defineProperty() 参考:https://segmentfault.com/a/1190000007434923 定义: 方法会直接在一个对象上定义一个新属性,或者修改一个 ...
- 使用fetch代替ajax请求 post传递方式
let postData = {a:'b'}; fetch('http://data.xxx.com/Admin/Login/login', { method: 'POST', mode: 'cors ...
- JS 获取链接中的参数
1.获取链接全部参数,以对象的形式返回 //获取url中参数 function GetRequest() { var url = location.search; //获取url中"?&qu ...
- composer 自动加载 通过classmap自动架子啊
https://github.com/brady-wang/composer github地址 composer加载自己写的类 放入一个目录下 更改composer.json "autolo ...
- [转帖]你所不知道的C和C++运行库
[C-C++]你所不知道的C和C++运行库 https://blog.csdn.net/humanking7/article/details/85887884 原作者也是转的blog 最近一个物理机上 ...
- .Net批量插入数据
1. 一般我们普通数据插入是这样的: 现在我们写一个控制台程序用常规办法添加10000条数据. //以下是批量插入数据的办法 //连接字符串 string str = "Server=.;D ...
- 将大数组里面的小数组平行展开的实现(Making a flat list out of list of lists in Python)
今天在生成数据的时候遇到了这个需求,其实写一个for循环可以很容易解决这个问题,但是无论是性能还是酷炫程度上都不行 所以顺手搜索了一下. 例子是将 l = [[1, 2, 3], [4, 5, 6], ...
- 关于Select2下拉框组件
文档如下: https://select2.org/configuration/options-api