因为最多有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. Thinkphp5的使用phpmailer实现发邮件功能(163邮箱)

    Thinkphp5本身并没有实现发邮件的功能,至少据我所知. 本文利用网易邮箱作为发邮件的邮箱.作为发送邮件的前提是需要开启SMTP服务,打开网易邮件,点击设置按钮,如下图所示 勾选smtp服务 保存 ...

  2. oracle怎样查询索引的使用情况

    查询用户的索引select index_name,table_name,tablespace_name, index_type,uniqueness , status from dba_indexes ...

  3. mysql学习之主从复制

    该文使用mysql5.5 centos6.5 64位 一.主从复制的作用 1.如果主服务器出现问题,可以快速切换到从服务器. 2.对与实时性要求不高或者更新不频繁的应用可以在从服务器上执行查询操作,降 ...

  4. 面试:谈谈你对Spring框架的理解

    Spring是一个优秀的轻量级框架,大大的提高了项目的开发管理与维护.Spring有两个核心模块.一个是IOC,一个是AOP. IOC: 就是控制反转的意思,指的是我们将对象的控制权从应用代码本身转移 ...

  5. Apple - Hdu5160

    Problem Description We are going to distribute apples to n children. Every child has his/her desired ...

  6. JS作用域-面向对象

    1. 其它语言是以代码块作为作用域的.下面程序会报错(如C,C++中),因为局部变量name只在{   }代码块中生效.打印console.writeline(name)中的name时就会报错. pu ...

  7. 【题解】CF#229 E-Gifts

    尽管是一道E题,但真心并不很难~不难发现,有一些物品是一定要被选择的,我们所需要决策的仅仅只有那几个有重复价值的物品. 而不同名字之间的概率并不互相影响,所以我们有 \(f[i][j]\) 表示名字为 ...

  8. [LOJ#2340] [WC2018] 州区划分

    题目链接 洛谷题面. LOJ题面.还是LOJ机子比较快 Solution 设\(f(s)\)表示选\(s\)这些城市的总代价,那么我们可以得到一个比较显然的\(dp\): \[ f(s)=\frac{ ...

  9. BZOJ3669:[NOI2014]魔法森林——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=3669 https://www.luogu.org/problemnew/show/P2387 为了得 ...

  10. [学习笔记]2-SAT 问题

    (本文语言不通,细节省略较多,不适合初学者学习) 解决一类简单的sat问题. 每个变量有0/1两种取值,m个限制条件都可以转化成形如:若x为0/1则y为0/1等等(x可以等于y) 具体: 每个变量拆成 ...