题目:http://uoj.ac/problem/348

一开始可以 3^n 子集DP,枚举一种状态的最后一个集合是什么来转移;

设 \( f[s] \) 表示 \( s \) 集合内的点都划分好了,\( g[s] = \sum\limits_{i \in s} w[i] \)

那么 \( f[s] = \sum\limits_{d \subseteq s} \frac{f[s-d] * g[d]}{g[s]} \)

注意判断一个集合是否合法,不仅要判断每个点的度数,还要判断整个集合是否连通;

这样就可以过 n <= 15 的点了,UOJ上有30分;

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const xn=(<<)+,xxn=,xm=,mod=;
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
int n,m,p,f[xn],g[xn],w[xxn],g2[xn];
int hd[xxn],ct,to[xm],nxt[xm],bin[xxn];
void add(int x,int y){to[++ct]=y; nxt[ct]=hd[x]; hd[x]=ct;}
int upt(int x){while(x>=mod)x-=mod; while(x<)x+=mod; return x;}
ll pw(ll a,int b){ll ret=; for(;b;b>>=,a=a*a%mod)if(b&)ret=ret*a%mod; return ret;}
bool vis[xxn];
int dfs(int x,int s)
{
vis[x]=; int ret=;
for(int i=hd[x],u;i;i=nxt[i])
if(!vis[u=to[i]]&&(s&bin[u-]))ret+=dfs(u,s);
return ret;
}
bool ck(int s)//
{
int cnt=;
for(int x=;x<=n;x++)
{
if(!(s&bin[x-]))continue;
int deg=; cnt++;
for(int i=hd[x];i;i=nxt[i])
{
if(s&bin[to[i]-])deg++;
}
if(deg&)return ;
}
for(int i=;i<=n;i++)vis[i]=;
for(int i=;i<=n;i++)
if(s&bin[i-])return dfs(i,s)!=cnt;
}
int main()
{
n=rd(); m=rd(); p=rd();
bin[]=; for(int i=;i<=n;i++)bin[i]=bin[i-]*;
for(int i=,x,y;i<=m;i++)x=rd(),y=rd(),add(x,y),add(y,x);
for(int i=;i<=n;i++)g[bin[i-]]=rd();
for(int s=;s<bin[n];s++)g[s]=upt(g[s&(-s)]+g[s^(s&(-s))]);
for(int s=;s<bin[n];s++)g[s]=pw(g[s],p),g2[s]=pw(g[s],mod-);
for(int s=;s<bin[n];s++)if(!ck(s))g[s]=;
int num=;
f[]=;
for(int s=;s<bin[n];s++)
{
for(int d=s;d;d=(s&(d-)))//d=s
f[s]=(f[s]+(ll)f[s^d]*g[d])%mod;
f[s]=(ll)f[s]*g2[s]%mod;
}
printf("%d\n",f[bin[n]-]);
return ;
}

3^n

关于FMT(其实和高维前缀和差不多)和子集卷积:https://www.cnblogs.com/Dance-Of-Faith/p/8818211.html

于是可以做子集卷积加速DP的过程。

代码如下:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const xn=(<<)+,xxn=,xm=,mod=;
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
int n,m,p,f[xxn][xn],g[xxn][xn],w[xxn],g2[xn];
int hd[xxn],ct,to[xm],nxt[xm],bin[xxn],cnt[xn];
void add(int x,int y){to[++ct]=y; nxt[ct]=hd[x]; hd[x]=ct;}
int upt(int x){while(x>=mod)x-=mod; while(x<)x+=mod; return x;}
ll pw(ll a,int b){ll ret=; for(;b;b>>=,a=a*a%mod)if(b&)ret=ret*a%mod; return ret;}
bool vis[xxn];
int dfs(int x,int s)
{
vis[x]=; int ret=;
for(int i=hd[x],u;i;i=nxt[i])
if(!vis[u=to[i]]&&(s&bin[u-]))ret+=dfs(u,s);
return ret;
}
bool ck(int s)//
{
int cnt=;
for(int x=;x<=n;x++)
{
if(!(s&bin[x-]))continue;
int deg=; cnt++;
for(int i=hd[x];i;i=nxt[i])
{
if(s&bin[to[i]-])deg++;
}
if(deg&)return ;
}
for(int i=;i<=n;i++)vis[i]=;
for(int i=;i<=n;i++)
if(s&bin[i-])return dfs(i,s)!=cnt;
}
int cal(int s){int ret=; while(s)ret+=(s&),s>>=; return ret;}
void fmt(int *a,int tp)
{
for(int d=;d<bin[n];d<<=)
for(int s=;s<bin[n];s++)
if(s&d)a[s]=upt(a[s]+a[s^d]*tp);
}
int main()
{
n=rd(); m=rd(); p=rd();
bin[]=; for(int i=;i<=n;i++)bin[i]=bin[i-]*;
for(int i=,x,y;i<=m;i++)x=rd(),y=rd(),add(x,y),add(y,x);
for(int s=;s<bin[n];s++)cnt[s]=cal(s);
for(int i=;i<=n;i++)g2[bin[i-]]=rd();
for(int s=;s<bin[n];s++)g2[s]=upt(g2[s&(-s)]+g2[s^(s&(-s))]);
for(int s=;s<bin[n];s++)g[cnt[s]][s]=pw(g2[s],p),g2[s]=pw(g[cnt[s]][s],mod-);
for(int s=;s<bin[n];s++)if(!ck(s))g[cnt[s]][s]=;
for(int i=;i<=n;i++)fmt(g[i],);
f[][]=; fmt(f[],);
for(int i=;i<=n;i++)
{
for(int j=;j<=i;j++)
for(int s=;s<bin[n];s++)
f[i][s]=(f[i][s]+(ll)f[j][s]*g[i-j][s])%mod;
fmt(f[i],-);
for(int s=;s<bin[n];s++)
if(cnt[s]==i)f[i][s]=(ll)f[i][s]*g2[s]%mod;
else f[i][s]=;
fmt(f[i],);
}
fmt(f[n],-);
printf("%d\n",f[n][bin[n]-]);
return ;
}

UOJ #348 州区划分 —— 状压DP+子集卷积的更多相关文章

  1. UOJ348 WC2018 州区划分 状压DP、欧拉回路、子集卷积

    传送门 应该都会判欧拉回路吧(雾 考虑状压DP:设\(W_i\)表示集合\(i\)的点的权值和,\(route_i\)表示点集\(i\)的导出子图中是否存在欧拉回路,\(f_i\)表示前若干个城市包含 ...

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

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

  3. 【UOJ348】【WC2018】州区划分 状压DP FWT

    题目大意 给定一个\(n\)个点的无向图,对于每种 \(n\) 个点的划分\(\{S_1,S_2,\ldots,S_k\}\),定义它是合法的,当且仅当每个点都在其中的一个集合中且对于任何的\(i\i ...

  4. HDU6321 Dynamic Graph Matching【状压DP 子集枚举】

    HDU6321 Dynamic Graph Matching 题意: 给出\(N\)个点,一开始没有边,然后有\(M\)次操作,每次操作加一条无向边或者删一条已经存在的边,问每次操作后图中恰好匹配\( ...

  5. [WC2018]州区划分(状压,子集卷积)

    [洛谷题面]https://www.luogu.org/problemnew/show/P4221 首先考虑判定一个子图是否合法: (1)连通:并查集判断即可. (2)没有欧拉回路:存在欧拉回路的条件 ...

  6. UOJ#348 州区划分

    解:有一个很显然的状压...... 就设f[s]表示选的点集为s的时候所有方案的权值和. 于是有f[s] = f[s \ t] * (sum[t] / sum[s])P. 这枚举子集是3n的. 然后发 ...

  7. 集合划分状压dp

    给一个 $n$ 个点 $m$ 条边的无向图,每条边有 $p_i$ 的概率消失,求图连通的概率 $n \leq 9$ sol: 我们考虑一个 $dp$ $f_{(i,S)}$ 表示只考虑前 $i$ 条边 ...

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

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

  9. 洛谷P3959 宝藏(NOIP2017)(状压DP,子集DP)

    洛谷题目传送门 Dalao的题解多数是什么模拟退火.DFS剪枝.\(O(3^nn^2)\)的状压DP之类.蒟蒻尝试着把状压改进了一下使复杂度降到\(O(3^nn)\). 考虑到每条边的贡献跟它所在的层 ...

随机推荐

  1. 九度OJ 1197:奇偶校验 (基础题)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:3590 解决:1511 题目描述: 输入一个字符串,然后对每个字符进行奇校验,最后输出校验后的二进制数(如'3',输出:10110011). ...

  2. Frobenius Norm

    http://mathworld.wolfram.com/FrobeniusNorm.html

  3. 【python】-- web开发之HTML

    HTML HTML是英文Hyper Text Mark-up Language(超文本标记语言)的缩写,是一种制作万维网页面标准语言(标记).通俗的讲就是相当于定义统一的一套规则,大家都来遵守他,这样 ...

  4. Linux软连接和硬连接

    软连接 命令: ln -s 原文件 目标文件 特征: 1.相当于windows的快捷方式 2.只是一个符号连接,所以软连接文件大小都很小 3.当运行软连接的时候,会根据连接指向找到真正的文件,然后执行 ...

  5. G20峰会将会给数字货币带来哪些影响?

    G20峰会对于全球经济有着举足轻重的影响,其成员人口占全球的2/3,国土面积占全球的60%,国内生产总值占全球的90%,贸易额占全球的75%……作为国际经济合作的主要平台,G20在引领和推动国际经济合 ...

  6. 第6条:在单次切片操作内,不要同时指定start、end和stride

    核心知识点: 1.使用负步进可以反转取值字符串及ASCII. 2.stride最好不要与start和end用在一起,会降低代码可读性. 除了基本的切片操作之外,python还提供了somelist[s ...

  7. 为什么下了android 4.1 的SDK后在本地用浏览器看api说明文档时,浏览器打开api的html文件很慢?试了好几款浏览器都一样。为什么?

    http://www.oschina.net/question/436724_61401 http://www.google.com/jsapi  他惹的祸 注释掉就可以了- <!-- < ...

  8. PAT 天梯赛 L2-007. 家庭房产 【并查集】

    题目链接 https://www.patest.cn/contests/gplt/L2-007 思路 将一个家庭里的所有人都并进去 然后最后查找的时候 找到所有同一个家庭的人,计算出人数,人均房产套数 ...

  9. UNIGUI:How to redirect and close session?

    Hello, i would have 2 unigui app. the first app is a simple authentification app and second will be ...

  10. LINQ 学习路程 -- 查询操作 Expression Tree

    表达式树就像是树形的数据结构,表达式树中的每一个节点都是表达式, 表达式树可以表示一个数学公式如:x<y.x.<.y都是一个表达式,并构成树形的数据结构 表达式树使lambda表达式的结构 ...