【WC2018】州区划分(FWT,动态规划)
【WC2018】州区划分(FWT,动态规划)
题面
题解
首先有一个暴力做法(就有\(50\)分了)
先\(O(2^nn^2)\)预处理出每个子集是否合法,然后设\(f[S]\)表示当前的答案,每次枚举一个子集进行转移,得到方程:\(\displaystyle f[S]=(\frac{1}{W_s})^p\sum_{T\subset S}f[T]*(W_{S-T})^p*check[S-T]\)。
其中\(W\)表示权值和,\(check\)表示是否合法。
这样子的复杂度是\(O(3^n)\),然后似乎可以拿\(50\)分了。
后面那个东西很像一个卷积,然而直接\(or\)卷积的话,会算出很多我们不想要的东西。
听说这个玩意叫做子集卷积。
要做的就是\(\sum f[T]*g[W]*[T\cup W=S,T\cap W=\phi]\)
可以用二进制下\(1\)的个数来表示这个限制\(\sum f[T]*g[W]*[T\cup W=S,cnt(T)+cnt(W)=cnt(S)]\)。
其中\(cnt(S)\)表示\(S\)中\(1\)的个数。
然后听说这个玩意就是一个套路了。。。
把集合的\(1\)的个数强制作为一维状态加上去。也就是\(f[cnt(S)][S]\)这样子。
令\(g[cnt(S)][S]=(W_S)^p*check[S]\)。
这样子的话只需要枚举两者的\(1\)的个数就可以去掉位的限制,然后就只剩下两者的交是\(S\)的限制,这个限制可以直接表示为或卷积(异或似乎也行???)。
那么直接\(FWT\)按层处理即可。
#include<iostream>
#include<cstdio>
using namespace std;
#define MOD 998244353
#define MAX 25
void add(int &x,int y){x+=y;if(x>=MOD)x-=MOD;}
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 n,m,p,S,G[MAX],dg[MAX],lg[1<<21];
bool chk[1<<21];
int cc[1<<21];
int f[22][1<<21],g[22][1<<21],tmp[1<<21];
int w[MAX],W[1<<21],Wk[1<<21],invW[1<<21];
void FWT(int *a,int opt)
{
for(int i=1;i<S;i<<=1)
for(int p=i<<1,j=0;j<S;j+=p)
for(int k=0;k<i;++k)
if(opt==1)add(a[i+j+k],a[j+k]);
else add(a[i+j+k],MOD-a[j+k]);
}
int fa[MAX];
int getf(int x){return x==fa[x]?x:fa[x]=getf(fa[x]);}
int main()
{
n=read();m=read();p=read();S=1<<n;
for(int i=2;i<S;++i)lg[i]=lg[i>>1]+1;
for(int i=1;i<=m;++i)
{
int u=read()-1,v=read()-1;
G[u]|=1<<v;G[v]|=1<<u;
}
for(int i=0;i<n;++i)w[i]=read();
for(int i=1;i<S;++i)cc[i]=cc[i>>1]+(i&1);
for(int i=0;i<S;++i)
{
for(int j=0;j<n;++j)fa[j]=j;
for(int j=0;j<n;++j)if(i&(1<<j))dg[j]=cc[G[j]&i];
for(int j=0;j<n;++j)
if(i&(1<<j))
{
int x=i&G[j];
while(x)fa[getf(j)]=getf(lg[x&(-x)]),x-=x&(-x);
}
chk[i]=false;int nw=0;
for(int j=0;j<n;++j)if(i&(1<<j))W[i]+=w[j],nw=j;
for(int j=0;j<n&&!chk[i];++j)if((dg[j]&1)||((i&(1<<j))&&dg[j]==0))chk[i]=true;
for(int j=0;j<n;++j)dg[j]=0;
for(int j=0;j<n&&!chk[i];++j)if(i&(1<<j))if(getf(j)!=getf(nw))chk[i]=true;
if(cc[i]==1)chk[i]=false;
Wk[i]=fpow(W[i],p);invW[i]=fpow(Wk[i],MOD-2);
if(chk[i])g[cc[i]][i]=Wk[i];
}
f[0][0]=1;FWT(f[0],1);
for(int i=1;i<=n;++i)
{
FWT(g[i],1);
for(int j=0;j<i;++j)
for(int k=0;k<S;++k)
add(tmp[k],1ll*f[j][k]*g[i-j][k]%MOD);
FWT(tmp,-1);
for(int j=0;j<S;++j)f[i][j]=1ll*tmp[j]*invW[j]%MOD,tmp[j]=0;
if(i!=n)FWT(f[i],1);
}
/*
f[0]=1;
for(int i=1;i<S;++i)
{
for(int t=(i-1)&i;;t=(t-1)&i)
{
if(chk[i^t])add(f[i],1ll*f[t]*Wk[i^t]%MOD);
if(!t)break;
}
f[i]=1ll*f[i]*invW[i]%MOD;
}
*/
printf("%d\n",f[n][S-1]);
return 0;
}
【WC2018】州区划分(FWT,动态规划)的更多相关文章
- [WC2018]州区划分——FWT+DP+FST
题目链接: [WC2018]州区划分 题目大意:给n个点的一个无向图,点有点权,要求将这n个点划分成若干个部分,每部分合法当且仅当这部分中所有点之间的边不能构成欧拉回路.对于一种划分方案,第i个部分的 ...
- [WC2018]州区划分(FWT)
题目描述 题解 这道题的思路感觉很妙. 题目中有一个很奇怪的不合法条件,貌似和后面做题没有什么关系,所以我们先得搞掉它. 也就是判断一个点集是否合法,也就是判断这个点集是否存在欧拉回路. 如果存在欧拉 ...
- [WC2018]州区划分(FWT,FST)
[WC2018]州区划分(FWT,FST) Luogu loj 题解时间 经典FST. 在此之前似乎用到FST的题并不多? 首先预处理一个子集是不是欧拉回路很简单,判断是否连通且度数均为偶数即可. 考 ...
- [WC2018]州区划分
[WC2018]州区划分 注意审题: 1.有序选择 2.若干个州 3.贡献是州满意度的乘积 枚举最后一个州是哪一个,合法时候贡献sum[s]^p,否则贡献0 存在欧拉回路:每个点都是偶度数,且图连通( ...
- [UOJ#348][WC2018]州区划分
[UOJ#348][WC2018]州区划分 试题描述 小 \(S\) 现在拥有 \(n\) 座城市,第ii座城市的人口为 \(w_i\),城市与城市之间可能有双向道路相连. 现在小 \(S\) 要将这 ...
- P4221 [WC2018]州区划分 无向图欧拉回路 FST FWT
LINK:州区划分 把题目中四个条件进行规约 容易想到不合法当前仅当当前状态是一个无向图欧拉回路. 充要条件有两个 联通 每个点度数为偶数. 预处理出所有状态. 然后设\(f_i\)表示组成情况为i的 ...
- Luogu4221 WC2018州区划分(状压dp+FWT)
合法条件为所有划分出的子图均不存在欧拉回路或不连通,也即至少存在一个度数为奇数的点或不连通.显然可以对每个点集预处理是否合法,然后就不用管这个奇怪的条件了. 考虑状压dp.设f[S]为S集合所有划分方 ...
- [WC2018]州区划分(状压DP+FWT/FMT)
很裸的子集反演模板题,套上一些莫名其妙的外衣. 先预处理每个集合是否合法,再作显然的状压DP.然后发现可以写成子集反演的形式,直接套模板即可. 子集反演可以看这里. 子集反演的过程就是多设一维代表集合 ...
- uoj#348/洛谷P4221 [WC2018]州区划分(FWT)
传送门(uoj) 传送门(洛谷) 全世界都会子集卷积就咱不会--全世界都在写\(FMT\)就咱只会\(FWT\)-- 前置芝士 或运算\(FWT\)或者\(FMT\) 左转洛谷模板区,包教包会 子集卷 ...
- [LOJ#2340] [WC2018] 州区划分
题目链接 洛谷题面. LOJ题面.还是LOJ机子比较快 Solution 设\(f(s)\)表示选\(s\)这些城市的总代价,那么我们可以得到一个比较显然的\(dp\): \[ f(s)=\frac{ ...
随机推荐
- Xcode中控制台中打印中文处理
xcode 10以后的方法,一般使用 #ifdef DEBUG #define NSLog(FORMAT, ...) fprintf(stderr,"\n %s:%d %s\n", ...
- Redis服务端的搭建(初级)
前方低能,仅适合入门级菜鸟阅读,大神大牛通通闪开! 前言:redis经常被用来做缓存(原因自行科普),基于学习的需要自己搭建了一个redis服务器,考虑到项目的分布式部署,所以前期开始的时候,redi ...
- 课程存储校对:程序设计思想、源程序代码、运行结果截图,以及开发过程中的项目计划日志、时间记录日志、缺陷记录日志(PSP0级记录)。
1.程序设计思想 ⑴将JDBC驱动jar包导入到WEB-INF的lib文件夹下 ⑵建立数据库,在数据库中建表,分别将课程名称.任课教师及上课地点录入到列中 ⑶首先写出加载驱动.关闭资源的工具类和异常处 ...
- shell脚本--编写CGI代码(shell结合html)以及环境变量
实现shell和html标签混合的方式编写代码: 推荐 初始CGI ,看完大概之后,大概对cgi有个大体的印象.下面是编写混合代码的示例: #!/bin/bash #index.cgi echo & ...
- Linux系统mysql使用(一)
一.安装 sudo apt-get update #更新软件源 sudo apt-get install mysql-server #安装mysql 二.启动和关闭 service mysql sta ...
- 【转】实现Nginx代理WSS协议
https://blog.csdn.net/chopin407/article/details/52937645 后来看到了官网的教程(http://nginx.org/en/docs/http/we ...
- PHP中友好的处理方式
在使用PHP进行开发的时候,由于PHP是弱类型语言的特性,所以,偶尔会遇到一些意想不到的错误.规范我们的编程就变得尤为重要了.下面总结一下,我日常开发中的一些经验,可能有些地方不妥,还请多多斧正,指教 ...
- HashMap深度解析(转载)
原文地址:http://blog.csdn.net/ghsau/article/details/16890151 实现原理:用一个数组来存储元素,但是这个数组存储的不是基本数据类型.HashMap实现 ...
- [转帖]Windows平台卸载Oracle的办法
1.首先打开服务:选中此电脑->点击右键->选择管理->选择服务和应用程序->服务 在右边查看并停止以 oracle开头的服务(选中正在运行的以oracle开头的服务-> ...
- ZJU_1145 OR POJ_1100 Dreisam Equations
Dreisam Equations { 两个网站的题有点不一样(ZJH有特判)POJ时间卡得紧,建议去POJ过 } 题目大意: 给你一个字符串:是一个等式,等式左边是一个数,右边由若干个数和()构成, ...