题目

分析

其实原题就是【cqoi2012】【bzoj2669】局部极小值。

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

给出所有局部极小值的位置,你的任务是判断有多少个可能的矩阵。

发现,X的位置最多有8个,那我们考虑状压dp。

我们从小到大把数填进去,用\(f_{i,j}\)表示,把第i个数填进去后,每个X是否被填了数,用二进制数j表示。

预处理出\(rest_j\)表示填充状态为j时共有多少位置是可以填充的(包括已填充的局部极小值位置)

转移:

\[f_{i,j}=f_{i-1,j}*(rest_j-(i-1))+\sum_{k\in{j}}f_{i-1,j-2^{k-1}}
\]

但是有些不是为X的位置有可能也是局部极小值,那么我们用容斥,每次把一下有可能出现局部极小值的地方改为X,当额外增加的X的个数为奇数,ans就减去dp得所得的答案,否则ans加上dp得所得的答案。

其中dp方面这篇论文(第5页到第8页)讲的非常清楚

#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const int maxlongint=2147483647;
const int mo=12345678;
const int N=10;
int z[9][2]=
{
{-1,-1},
{-1,0},
{-1,1},
{0,-1},
{0,1},
{1,-1},
{1,0},
{1,1},
{0,0}
};
int a[N][N],T,n,m,ans,f[N*N][2000],mi[10],sign[N][2],tot,rest[2000];
char c[N][N];
bool bz[N][N];
int val()
{
tot=0;
int state=0;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
{
if(c[i][j]=='X')
{
state+=mi[tot];
sign[++tot][0]=i;
sign[tot][1]=j;
}
}
for(int i=0;i<=state;i++)
{
memset(bz,true,sizeof(bz));
rest[i]=0;
for(int j=1;j<=tot;j++)
if((mi[j-1]&i)==0)
{
for(int k=0;k<=8;k++)
{
bz[sign[j][0]+z[k][0]][sign[j][1]+z[k][1]]=false;
}
}
for(int j=1;j<=n;j++)
for(int k=1;k<=m;k++)
{
if(bz[j][k])
rest[i]++;
}
}
f[0][0]=1;
for(int i=1;i<=n*m;i++)
for(int j=0;j<=state;j++)
{
f[i][j]=0;
(f[i][j]+=f[i-1][j]*(rest[j]-i+1)%mo)%=mo;
for(int k=1;k<=tot;k++)
{
if(mi[k-1]&j)
{
(f[i][j]+=f[i-1][j-mi[k-1]])%=mo;
}
}
}
return f[n*m][mi[tot]-1];
}
int dg(int x,int y,int z1)
{
if(x>n)
{
(ans+=(val()*(z1%2?1:-1))%mo)%mo;
return 0;
}
int xx=x,yy=y+1;
if(yy>m)
{
yy=1;
xx++;
}
dg(xx,yy,z1);
bool q=true;
for(int i=0;i<=8;i++)
{
if(c[x+z[i][0]][y+z[i][1]]=='X')
{
q=false;
break;
}
}
if(q)
{
c[x][y]='X';
dg(xx,yy,z1+1);
c[x][y]='.';
}
}
int main()
{
mi[0]=1;
for(int i=1;i<=9;i++)
mi[i]=mi[i-1]*2;
scanf("%d",&T);
while(T--)
{
tot=0;
ans=0;
memset(c,0,sizeof(c));
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
c[i][j]=getchar();
while(c[i][j]!='X' && c[i][j]!='.')
c[i][j]=getchar();
if(c[i][j]=='.')
tot++;
}
}
if(tot==n*m)
{
printf("0\n");
continue;
}
for(int i=1;i<=n && ans!=-1;i++)
{
for(int j=1;j<=m;j++)
{
bool q=true;
if(c[i][j]=='X')
for(int k=0;k<=7;k++)
{
if(c[i+z[k][0]][j+z[k][1]]=='X')
{
q=false;
ans=-1;
break;
}
}
if(!q)
{
ans=-1;
break;
}
}
}
if(ans==-1)
{
printf("0\n");
}
if(ans!=-1)
{
dg(1,1,1);
printf("%d\n",(ans%mo+mo)%mo);
}
}
}

【NOIP2016提高A组模拟8.15】Garden的更多相关文章

  1. NOIP2016提高A组模拟10.15总结

    第一题,就是将原有的式子一步步简化,不过有点麻烦,搞了很久. 第二题,枚举上下边界,维护一个单调队列,二分. 比赛上没有想到,只打了个暴力,坑了80分. 第三题,贪心,最后的十多分钟才想到,没有打出来 ...

  2. 【NOIP2016提高A组模拟10.15】打膈膜

    题目 分析 贪心, 先将怪物按生命值从小到大排序(显然按这个顺序打是最优的) 枚举可以发对少次群体攻击, 首先将所有的群体攻击发出去, 然后一个一个怪物打,当当前怪物生命值大于2,如果还有魔法值就放重 ...

  3. 【NOIP2016提高A组模拟10.15】最大化

    题目 分析 枚举两个纵坐标i.j,接着表示枚举区域的上下边界, 设对于每个横坐标区域的前缀和和为\(s_l\),枚举k, 显然当\(s_k>s_l\)时,以(i,k)为左上角,(j,k)为右下角 ...

  4. 【NOIP2016提高A组模拟10.15】算循环

    题目 分析 一步步删掉循环, 首先,原式是\[\sum_{i=1}^n\sum_{j=1}^m\sum_{k=i}^n\sum_{l=j}^m\sum_{p=i}^k\sum_{q=j}^l1\] 删 ...

  5. 【NOIP2016提高A组模拟9.15】Map

    题目 分析 发现,当原图是一棵树的时候,那么新建一条边后,就会变成环套树, 而环内的所有点对都是安全点对,如果环中有k个点,答案就是\(k(k-1)\) 联想到,当把原图做一遍tarjan缩点,每个环 ...

  6. 【NOIP2016提高A组模拟9.15】Osu

    题目 分析 考虑二分答案, 二分小数显然是不可取的,那么我们将所有可能的答案求出来,记录在一个数组上,排个序(C++调用函数很容易超时,手打快排,时间复杂度约为\(O(>8*10^7)\),但相 ...

  7. 【NOIP2016提高A组模拟9.15】Math

    题目 分析 因为\((-1)^2=1\), 所以我们只用看\(\sum_{j=1}^md(i·j)\)的值模2的值就可以了. 易证,一个数x,只有当x是完全平方数时,d(x)才为奇数,否则为偶数. 那 ...

  8. 【NOIP2016提高A组模拟8.15】Throw

    题目 分析 首先对于一个状态(a,b,c),假定a<=b<=c: 现在考虑一下这个状态,的转移方案: \[1,中间向两边跳(a,b,c)-->(a*2-b,a,c).(a,b,c)- ...

  9. 【NOIP2016提高A组模拟8.15】Password

    题目 分析 首先我们知道,原A序列其实表示一个矩阵,而这个矩阵的对角线上的数字就是答案B序列. 接着\(a.b>=gcd(a,b)\),所以序列A中的最大的数就是ans[1],第二大的数就是an ...

随机推荐

  1. 【HANA系列】SAP HANA使用XS和HTTP创建proxy

    公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[HANA系列]SAP HANA使用XS和HTT ...

  2. 配置本地访问远程Linux系统服务器的jupyter notebook

    环境情况 远程服务器上配置了anaconda 本地主机没有安装anaconda(其实安不安装都无所谓,有浏览器就行) 配置步骤如下 登录远程服务器 生成配置文件 jupyter notebook -- ...

  3. C++中内联函数的用法

    程序带调用函数需要一定的时间\空间花销,这就要求在主程序进行过程中调用函数前几下执行指令的地址及其他相关信息,一边函数调用后能继续执行.函数调用后流程返回先前记下的地址处,并根据记录的相关信息回复,而 ...

  4. lua基础学习(五)

    一.Lua 模块与包 模块类似于一个封装库,从 Lua 5.1 开始,Lua 加入了标准的模块管理机制,可以把一些公用的代码放在一个文件里,以 API 接口的形式在其他地方调用,有利于代码的重用和降低 ...

  5. CoolFormat源代码格式化工具(转)

    软件介绍: CoolFormat源代码格式化是一款C\C++\C#\CSS\HTML\Java\JavaScript\JSON\Objective-C\PHP\SQL\XML代码格式化工具.软件可以快 ...

  6. Java 创建bat命令文件运行可执行jar包

    在可执行jar包所在文件夹下创建txt文件(必须在同一文件夹目录下),打开创建的txt文件输入如下内容并保存: @echo off java -jar 包名.jar pause 如下图所示: 然后将后 ...

  7. rk3288 编译应用程序

    一. Android.mk 1.1. 什么是.mk Android.mk是Android提供的一个makefile文件,可以将源文件分组为模块.用来引用的头文件目录.需要编译的*.c/*.cpp文件. ...

  8. C中的异常处理

    1,C 语言崇尚简洁高效,因此语言本身并没有异常处理的相关语法规则,但是异常处理在 C 语言中 是存在的,我们有必要从 C 语言开始先看一看 C 语言中的异常处理是怎样, 然后对比 C++ 里面的异常 ...

  9. Head First PHP&MySQl第二章代码

    PHP: <html> <head> <title>外星人绑架了我--报道一起绑架</title> </head> <body> ...

  10. ubuntu18下lamp虚拟路劲配置

    一.配置二级域名 修改hosts文件,模拟dns解析. 位置:/etc/hosts 添加 127.0.0.1  myweb.service.com 二.创建项目目录 apache默认目录是/var m ...