首页 > 其他 > 详细

bzoj2560串珠子——子集DP

时间:2018-06-05 00:03:48      阅读:265      评论:0      收藏:0      [点我收藏+]

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2560

转载:


很明显的状压dp 
一开始写的dp可能会出现重复统计的情况 而且难以去重 
假设 一个状态s的随意连边集合是A; 
那么 A应该是 全部合法的方案(Ans)+sigma(某一部分合法(即某一部分是连通图)的方案*其他任意连边的方案); 
那么可以把最终答案设置为f[i], 随意连边(也可以完全连边)设置成g[i]; 
先定一个基准点 x 和基准点相连的都是合法的, 其余集合 t=s^(1<<(x-1))可以随便连; 
f[i]=g[i]-sigma((t的所有子集i)f[i]*g[s^i]); 
为什么一个是f 一个是g 这样其实是要保证不重不漏


而且必须注意划分点一定是连通部分的,否则样例输出49...不知道为什么。

代码如下:

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
typedef long long ll;
ll const mod=1000000007;
ll n,a[20][20],f[1<<16],g[1<<16];
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++)
            scanf("%d",&a[i][j]);
    int m=(1<<n)-1;
    for(int s=1;s<=m;s++)
    {
        g[s]=1;
        for(int i=1;i<=n;i++)   if(s&(1<<(i-1)))
            for(int j=i+1;j<=n;j++)   if(s&(1<<(j-1)))
                (g[s]*=(a[i][j]+1))%=mod;
        f[s]=g[s];
        int nw;
        for(nw=n-1;;nw--)
            if(s&(1<<nw))break;
        nw=(s^(1<<nw));//去掉一个划分点 
        for(int k=nw;k;k=((k-1)&nw))//枚举nw的子集
            ((f[s]-=g[k]*f[s^k])+=mod)%=mod;//f和g别反 
    }
    printf("%lld",f[m]);
    return 0;
}

 

bzoj2560串珠子——子集DP

原文:https://www.cnblogs.com/Zinn/p/9136595.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!