因为最多有8个'X',所以我们可以用w[i][s]来表示现在我们填了前i个数,填的X的为S,因为每次新加进来的数都不影响前面的最小值,所以我们可以随便添加,这样就有了剩下所有位置的方案,每次都这样转移。

  但是这样会造成不是规定的地方出现局部最小值的情况,对于这样的情况,我们只需要枚举所有可能成为局部最小值的不合法状态来做容斥就可以了。

  反思:这道题的容斥开始写错了,本来应该是判奇偶来判断正负,写成了全是负的,还是A了,应该是后面的容斥没有合法方案所以符号无所谓的关系,真是rp++。

/**************************************************************
    Problem: 2669
    User: BLADEVIL
    Language: C++
    Result: Accepted
    Time:0 ms
    Memory:928 kb
****************************************************************/
 
//By BLADEVIL
#include <cstdio>
#include <cstring>
#define d39 12345678
#define get(x) ((x&1)?-1:1)
 
using namespace std;
 
char s[][];
int map[][],flag[][],w[][],X[],Y[],tot[],tmp[][];
int ans,sum,n,m;
const int go[][]={{-,-},{-,},{-,},{,},{,},{,},{,-},{,-}};
 
int solve() {
    int cnt=,x,y,flag;
    for (int i=;i<=n;i++)
        for (int j=;j<=m;j++) if (map[i][j]) X[cnt]=i,Y[cnt++]=j;
    for (int p=;p<(<<cnt);p++) {
        tot[p]=;
        memset(tmp,,sizeof tmp);
        for (int i=;i<cnt;i++) if (p&(<<i)) tmp[X[i]][Y[i]]=;
        for (int i=;i<=n;i++)
            for (int j=;j<=m;j++) if (!tmp[i][j]) {
                flag=;
                for (int k=;k<&&flag;k++) {
                    x=i+go[k][],y=j+go[k][];
                    if (x<||x>n||y<||y>m) continue;
                    if (tmp[x][y]) flag=;
                }
                if (flag) tot[p]++;
        }
    }
    memset(w,,sizeof w);
    w[][]=;
    int opp;
    for (int i=;i<=sum;i++) {
        for (int p=;p<(<<cnt);p++) if (w[i-][p]) { 
            opp=;
            for (int j=;j<cnt;j++) if (!(p&(<<j)))
                opp|=(<<j),(w[i][p|(<<j)]+=w[i-][p])%=d39;
            (w[i][p]+=w[i-][p]*(tot[opp]-(i-))%d39)%=d39;
        }
   }
   return w[sum][(<<cnt)-];
}
 
void ie(int x,int y,int t){
    map[x][y]=;
    int i,j,ret;
    for (int k=;k<;k++) {
        i=x+go[k][],j=y+go[k][];
        if (i<||i>n||j<||j>m) continue;
        if (flag[i][j]) continue;
        flag[i][j]=t;
    }
    ret=solve();
    ans=(ans+get(t)*ret)%d39;
    for (j=y+;j<=m;j++) if (flag[x][j]==) ie(x,j,t+);
    for (i=x+;i<=n;i++)
        for (j=;j<=m;j++) if (flag[i][j]==) ie(i,j,t+);
    map[x][y]=;
    for (int k=;k<;k++) {
        i=x+go[k][],j=y+go[k][];
        if (i<||i>n||j<||j>m) continue;
        if (flag[i][j]==t) flag[i][j]=;
    }
}
 
int main(){
    scanf("%d%d",&n,&m);
    int cur=,x,y;
    sum=n*m;
    for (int i = ;i <= n;i ++) {
        scanf("%s",s[i]);
        for (int j=;j<m;j++)map[i][j+]=s[i][j]=='X'?:;
    }
    for (int i=;i<=n&&cur;i++) {
        for (int j=;j<=m&&cur;j++) if (map[i][j]) {
            if (flag[i][j]==-) cur=;
            flag[i][j]=-;
            for (int k=;k<;k++) {
               x=i+go[k][],y=j+go[k][];
               if (x<||x>n||y<||y>m) continue;
               flag[x][y]=-;
           }
       }
    }
    if (!flag) {
        printf("0\n");
        return ;
    }
    ans=solve();
    for (int i=;i<=n;i++)
        for (int j=;j<=m;j++) if (flag[i][j]==) ie(i,j,);
    (ans+=d39)%=d39;
    printf("%d\n",ans);
    return ;
}

bzoj 2669 状压DP的更多相关文章

  1. bzoj 1879 状压dp

    879: [Sdoi2009]Bill的挑战 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 852  Solved: 435[Submit][Status ...

  2. bzoj 1087 状压dp

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4130  Solved: 2390[Submit][ ...

  3. BZOJ 2064 - 状压DP

    传送门 题目大意: 给两个数组, 数组中的两个元素可以合并成两元素之和,每个元素都可以分裂成相应的大小,问从数组1变化到数组2至少需要多少步? 题目分析: 看到数据范围\(n<=10\), 显然 ...

  4. BZOJ 4057 状压DP

    思路: 状压一下 就完了... f[i]表示选了的集合为i 转移的时候判一判就好了.. //By SiriusRen #include <cstdio> #include <cstr ...

  5. BZOJ 4565 状压DP

    思路: f[i][j][S]表示从i到j压成S状态 j-m是k-1的倍数 $f[i][j][S<<1]=max(f[i][j][S<<1],f[i][m-1][S]+f[m][ ...

  6. bzoj 1072状压DP

    1072: [SCOI2007]排列perm Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2293  Solved: 1448[Submit][St ...

  7. bzoj 1072 状压DP

    我们用w[i][j]来表示,i是一个二进制表示我们选取了s中的某些位,j表示这些位%d为j,w[i][j]则表示这样情况下的方案数,那么我们可以得到转移.w[i|(1<<k)][(j*10 ...

  8. bzoj 1076 状压DP

    我们设w[i][s]为当前到第i关,手中的物品为s的时候,期望得分为多少,其中s为二进制表示每种物品是否存在. 那么就比较容易转移了w[i][s]=(w[i-1][s']+v[j]) *(1/k),其 ...

  9. BZOJ 1231 状压DP

    思路: f[i][j] i表示集合的组成 j表示选最后一个数 f[i][j]表示能选的方案数 f[i|(1<< k)][k]+=f[i][j]; k不属于i j属于i且符合题意 最后Σf[ ...

随机推荐

  1. cURL和file_get_contents实现模拟post请求

    以前面试时候,面试官问过我后端有没有跨域问题,但是不敢肯定,现在可以肯定的说没有. 不文用php的cURL和file_get_contents方法分别实现后端跨域.本文场景也是在tp5下实现的. 一, ...

  2. SQL SERVER 存储过程中SELECT 返回值如何赋值给变量

    今天在处理一个问题时,使用到一个存储过程,是用于更新并获取最新ID的.在使用过程中,需要获取到这个ID并赋值给变量,结果用EXEC @ID = 存储过程的方式获取失败了.具体情况如下: 为了还原整个情 ...

  3. Binding自动侦听

    WPF的强大之一就是数据绑定,Binding是数据桥梁,它的两端是分别是源(Source)和目标(Target),一个简单的类的属性值发生变化,会自动反映在UI界面上,这个属性就是Binding的Pa ...

  4. Windows下BMP位图格式介绍

    BMP图片,是Bitmap(位图)的简称,它是windows下显示图片的基本格式.在windows下任何格式的图片文件(包括视频播放)都要转化为位图才能显示出来.各种格式的图片文件也都是在位图格式的基 ...

  5. 使用Visual C ++和Open Folder自定义环境

    使用Visual C ++和Open Folder自定义环境 来源 https://blogs.msdn.microsoft.com/vcblog/2016/10/05/bring-your-c-co ...

  6. [洛谷P4688][Ynoi2016]掉进兔子洞

    题目大意:给定一个$n(n\leqslant10^5)$序列,$m(m\leqslant10^5)$个询问,每个询问给出$l_1,r_1,l_2,r_2,l_3,r_3$.令$s$为该三个区间的交集的 ...

  7. POJ1389:Area of Simple Polygons——扫描线线段树题解+全套代码注释

    http://poj.org/problem?id=1389 题面描述在二维xy平面中有N,1 <= N <= 1,000个矩形.矩形的四边是水平或垂直线段.矩形由左下角和右上角的点定义. ...

  8. BZOJ3195:[JXOI2012]奇怪的道路——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=3195 Description 小宇从历史书上了解到一个古老的文明.这个文明在各个方面高度发达,交通方 ...

  9. 20165218 实验一 Java开发环境的熟悉

    实验一 Java开发环境的熟悉 课程:java程序设计 姓名:赵冰雨 学号:20165218 指导教师:娄嘉鹏 实验日期:2018.4.2 实验密级:Java开发环境的熟悉 实验内容.步骤与体会: ( ...

  10. Shell编程语法

    创建shell程序的步骤: 第一步:创建一个脚本文件.sh. 第二步:授予权限使它可以执行chmod u+x .sh 第三步:执行 ./example 或者 sh example 脚本调试:      ...