题目链接

注意到\(4\times 7\)的矩阵的局部极小值最多只有8个,可以状压。

设\(f[i][sta]\)表示从小到大填数,当前填到\(i\),极小值的填充状态为\(sta\)的方案数。

考虑到由于是从小到大填数,每个极小值位置没填时,其四周的位置也不能填。

那么可以分情况讨论:

当前填在极小值位置上,\(f[i][sta]+=f[i-1][sta-(1<<j)]\).其中\(j\)为枚举到的位置。

否则,处理出当前状态下可以填的位置数\(num\),以前填了\(i-1\)个,转移:\(f[i][sta]+=f[i-1][sta]*(num-(i-1))\).

最后的答案为\(f[n*m][(1<<num)-1]\),\(num\)为极小值的个数。

但是,可以发现,题目要求极小值只有输入的那几个,但是这样转移可能会在随便填的地方产生新的极小值。

拿这个答案减去多了1个极小值的方案,就可以排除多一个极小值的错误,但是发现多两个的多减了一个,在加上……

其实就是容斥了一下,即原答案\(-\)多了一个的方案\(+\)多了两个的方案.....

然后多了几个的方案可以暴力枚举多了哪几个,添加到原图中,然后和上面一样的dp出来。

#include<bits/stdc++.h>
using namespace std;
#define int long long void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=(x<<1)+(x<<3)+(ch^48);x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar((x%10)^48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} const int dx[]={0,1,1,1,0,-1,-1,-1,0};
const int dy[]={0,1,0,-1,1,1,0,-1,-1};
const int mod = 12345678; int n,m,mp[10][10],f[30][(1<<9)+10],ans,vx[10],vy[10],num[1000],c[5][8];
char s[10]; int check(int x,int y) {return x>=1&&x<=n&&y>=1&&y<=m;} int calc() {
memset(f,0,sizeof f);f[0][0]=1;vx[0]=0,vy[0]=0;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(mp[i][j]) vx[++vx[0]]=i,vy[++vy[0]]=j;
for(int s=0;s<(1<<vx[0]);s++) {
memset(c,0,sizeof c);
for(int i=1;i<=vx[0];i++)
if(!(s&(1<<(i-1)))) {
for(int x,y,k=1;k<=8;k++)
if(check(x=vx[i]+dx[k],y=vy[i]+dy[k])) c[x][y]=1;
c[vx[i]][vy[i]]=1;
}num[s]=0;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
num[s]+=c[i][j]^1;
}
for(int i=1;i<=n*m;i++)
for(int s=0;s<(1<<vx[0]);s++) {
(f[i][s]+=f[i-1][s]*max(0ll,num[s]-(i-1))%mod)%=mod;
for(int j=1;j<=vx[0];j++)
if(s&(1<<(j-1))) f[i][s]=(f[i][s]+f[i-1][s^(1<<(j-1))])%mod;
}
return f[n*m][(1<<vx[0])-1];
} //void debug() {for(int i=1;i<=n;i++,puts("")) for(int j=1;j<=m;j++,putchar(' ')) printf("%d",mp[i][j]);} void dfs(int x,int y,int f) {
if(y==m+1) return dfs(x+1,1,f),void();
if(x==n+1) return ans=(ans+calc()*f)%mod,/*debug()*/void();
dfs(x,y+1,f);int flag=0;
for(int i=1;i<=8;i++)
if(check(x+dx[i],y+dy[i])&&mp[x+dx[i]][y+dy[i]]) {flag=1;break;}
if(mp[x][y]) flag=1;
if(!flag) mp[x][y]=1,dfs(x,y+1,-f),mp[x][y]=0;
} void solve () {
read(n),read(m);
for(int i=1;i<=n;i++) {
scanf("%s",s+1);//cout<<s+1<<endl;
for(int j=1;j<=m;j++) if(s[j]=='X') mp[i][j]=1;
}//debug();puts("");
dfs(1,1,1);write((ans%mod+mod)%mod);
} signed main() {solve();return 0;}

[CQOI2012]局部极小值的更多相关文章

  1. bzoj 2669 [cqoi2012]局部极小值 DP+容斥

    2669: [cqoi2012]局部极小值 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 838  Solved: 444[Submit][Status ...

  2. bzoj2669[cqoi2012]局部极小值 容斥+状压dp

    2669: [cqoi2012]局部极小值 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 774  Solved: 411[Submit][Status ...

  3. [BZOJ2669] [cqoi2012]局部极小值

    [BZOJ2669] [cqoi2012]局部极小值 Description 有一个n行m列的整数矩阵,其中1到nm之间的每个整数恰好出现一次.如果一个格子比所有相邻格子(相邻是指有公共边或公共顶点) ...

  4. 【BZOJ 2669】 2669: [cqoi2012]局部极小值 (状压DP+容斥原理)

    2669: [cqoi2012]局部极小值 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 667  Solved: 350 Description 有一 ...

  5. P3160 [CQOI2012]局部极小值

    题目 P3160 [CQOI2012]局部极小值 一眼就是状压,接下来就不知道了\(qwq\) 做法 我们能手玩出局部小值最多差不多是\(8,9\)个的样子,\(dp_{i,j}\)为填满\(1~i\ ...

  6. P3160 [CQOI2012]局部极小值 题解(状压DP+容斥)

    题目链接 P3160 [CQOI2012]局部极小值 双倍经验,双倍快乐 解题思路 存下来每个坑(极小值点)的位置,以这个序号进行状态压缩. 显然,\(4*7\)的数据范围让极小值点在8个以内(以下示 ...

  7. BZOJ2669 [cqoi2012]局部极小值 状压DP 容斥原理

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2669 题意概括 有一个n行m列的整数矩阵,其中1到nm之间的每个整数恰好出现一次.如果一个格子比所 ...

  8. 【bzoj2669】[cqoi2012]局部极小值 容斥原理+状压dp

    题目描述 有一个n行m列的整数矩阵,其中1到nm之间的每个整数恰好出现一次.如果一个格子比所有相邻格子(相邻是指有公共边或公共顶点)都小,我们说这个格子是局部极小值. 给出所有局部极小值的位置,你的任 ...

  9. BZOJ 2669 CQOI2012 局部极小值 状压dp+容斥原理

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2669 题意概述:实际上原题意很简洁了我就不写了吧.... 二话不说先观察一下性质,首先棋盘 ...

随机推荐

  1. webpack4提升180%编译速度

    前言 对于现在的前端项目而言,编译发布几乎是必需操作,有的编译只需要几秒钟,快如闪电,有的却需要10分钟,甚至更多,慢如蜗牛.特别是线上热修复时,分秒必争,响应速度直接影响了用户体验,用户不会有耐心等 ...

  2. webpack和sass功能简介

    1.webpack webpack 是一个打包工具,为什么需要打包?因为有的人的脚本开发语言可能是 CoffeeScript 或者是 TypeScript,样式开发工具可能是 Less 或者 Sass ...

  3. YII2 不通过composer安装Ueditor编辑器

    今天用composer安装Ueditor,一直下载失败,不知道为什么,所以就手动安装了一下.记录一下安装步骤 GitHub地址 https://github.com/BigKuCha/yii2-ued ...

  4. 廖老师的Python教程——Python简介

    一直想了解下Python,今儿在外面办事排队的时候,打开了廖老师的官网,找到了Python教程.虽然只是一篇关于Python的简介,但是通过将Python的特性与C进行对比,很生动地归纳了Python ...

  5. ruby URI类

    一. URI require 'uri' uri = URI("http://foo.com/posts?id=30&limit=5#time=1305298413") # ...

  6. Java学习笔记十:Java的数组以及操作数组

    Java的数组以及操作数组 一:什么是数组: 数组可以理解为是一个巨大的“盒子”,里面可以按顺序存放多个类型相同的数据,比如可以定义 int 型的数组 scores 存储 4 名学生的成绩 数组中的元 ...

  7. HDU暑假多校第八场J-Taotao Picks Apples

    一.题意 给定一个序列,之后给出若干个修改,修改的内容为在原序列的基础上,将某一位元素的值改成给定的值<每次修改相互独立,不保存修改后的结果>.之后询问,在选择第一位元素的情况下,最长递增 ...

  8. Ubuntu14.04安装opencv2.4.13

    本文参考相关链接:http://blog.csdn.net/honyniu/article/details/46390097   系 统:Ubuntu 14.04 x64 opencv版本:2.4.1 ...

  9. python2.7练习小例子(十二)

        12):题目:打印出所有的"水仙花数",所谓"水仙花数"是指一个三位数,其各位数字立方和等于该数本身.例如:153是一个"水仙花数" ...

  10. PHP.39-扩展-锁机制解决并发-MySQL锁、PHP文件锁

    锁机制适用于高并发场景:高并发订单.秒杀…… apache压力测试 Mysql锁详解 语法 加锁:LOCK TABLE 表名1 READ|WRITE, 表名2 READ|WRITE ......... ...