bzoj 2669 状压DP
因为最多有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的更多相关文章
- bzoj 1879 状压dp
879: [Sdoi2009]Bill的挑战 Time Limit: 4 Sec Memory Limit: 64 MBSubmit: 852 Solved: 435[Submit][Status ...
- bzoj 1087 状压dp
1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4130 Solved: 2390[Submit][ ...
- BZOJ 2064 - 状压DP
传送门 题目大意: 给两个数组, 数组中的两个元素可以合并成两元素之和,每个元素都可以分裂成相应的大小,问从数组1变化到数组2至少需要多少步? 题目分析: 看到数据范围\(n<=10\), 显然 ...
- BZOJ 4057 状压DP
思路: 状压一下 就完了... f[i]表示选了的集合为i 转移的时候判一判就好了.. //By SiriusRen #include <cstdio> #include <cstr ...
- 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][ ...
- bzoj 1072状压DP
1072: [SCOI2007]排列perm Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2293 Solved: 1448[Submit][St ...
- bzoj 1072 状压DP
我们用w[i][j]来表示,i是一个二进制表示我们选取了s中的某些位,j表示这些位%d为j,w[i][j]则表示这样情况下的方案数,那么我们可以得到转移.w[i|(1<<k)][(j*10 ...
- bzoj 1076 状压DP
我们设w[i][s]为当前到第i关,手中的物品为s的时候,期望得分为多少,其中s为二进制表示每种物品是否存在. 那么就比较容易转移了w[i][s]=(w[i-1][s']+v[j]) *(1/k),其 ...
- BZOJ 1231 状压DP
思路: f[i][j] i表示集合的组成 j表示选最后一个数 f[i][j]表示能选的方案数 f[i|(1<< k)][k]+=f[i][j]; k不属于i j属于i且符合题意 最后Σf[ ...
随机推荐
- 面试中常用排序算法的python实现和性能分析
这篇是关于排序的,把常见的排序算法和面试中经常提到的一些问题整理了一下.这里面大概有3个需要提到的问题: 虽然专业是数学,但是自己还是比较讨厌繁琐的公式,所以基本上文章所有的逻辑,我都尽可能的用大白话 ...
- WPF数据视图学习
当你绑定集合到ItemsControl,数据视图被安静地在幕后创造.视图位于数据源和绑定控件之间.数据视图是通往数据源的一个窗口.它跟踪当前项目,它支持诸如排序,过滤,和分组特征.这些特征独立于数据对 ...
- 【转】C# 利用反射动态创建对象
http://www.cnblogs.com/Jan_Dai/archive/2010/11/09/1872812.html Activator.CreateInstance(Type.GetType ...
- 【题解】HNOI2018转盘
何学长口中所说的‘一眼题’……然而实际上出出来我大HN全省也只有一个人A…… 首先我们需要发现一个性质:我们永远可以在最后一圈去标记所有的物品.倘若我们反复转圈,那么这完全是可以省下来的.所以我们破环 ...
- [COGS2652]秘术「天文密葬法」
description 题面 给个树,第\(i\)个点有两个权值\(a_i\)和\(b_i\),现在求一条长度为\(m\)的路径,使得\(\frac{\sum a_i}{\sum b_i}\)最小 d ...
- [洛谷P4168][Violet]蒲公英
题目大意:有$n(n\leqslant4\times10^4)$个数,$m(m\leqslant5\times10^4)$个询问,每次问区间$[l,r]$内的众数,若相同输出最小的,强制在线. 题解: ...
- [洛谷P2447][SDOI2010]外星千足虫
题目大意:有$n$个数,每个数为$0$或$1$,给你其中一些关系,一个关系形如其中几个数的异或和是多少,问最少知道前几个关系就可以得出每个数是什么,并输出每个数 题解:异或方程组,和高斯消元差不多,就 ...
- 【BZOJ5288】[HNOI2018]游戏(乱搞?)
[BZOJ5288][HNOI2018]游戏(乱搞?) 题面 BZOJ 洛谷 题面自己到洛谷上看把 题解 考场上乱搞拿到了\(90\)分,简直不敢相信. 回家把代码再交了一份直接就\(AC\)了??? ...
- 分享几款常用的API/文档浏览器
1.Dash 支持平台:Mac iOS 官网:https://kapeli.com/dash 2.Zeal 支持平台:Linux Windows 官网:https://zealdocs.org/ G ...
- [Leetcode] maximun subarray 最大子数组
Find the contiguous subarray within an array (containing at least one number) which has the largest ...