Solution -「Code+#2」「洛谷 P4033」白金元首与独舞
\(\mathcal{Description}\)
link.
给定一个 \(n\times m\) 的网格图,一些格子指定了走出该格的方向(上下左右),而有 \(k\) 格可以任意指定走出方向。求指定的方案数,使得从任意格子都可以走出网格图。
\(n,m\le200;k\le300\)。
\(\mathcal{Solution}\)
令“走出边界”为走到一个特殊点,建图,其中未指定方向的点向四周连边,相当于求以特殊点为根的内向树个数,跑矩阵树定理即可。复杂度 \(\mathcal O(n^3m^3)\)。
考虑优化,生成树个数实质上只与不定向的点有关,所以直接预处理出每个不定向点向上/下/左/右走到的第一个不定向点,向其连边,再跑矩阵树。复杂度 \(\mathcal O(k^3)\)。
\(\mathcal{Code}\)
#include <cstdio>
#include <cstring>
#include <iostream>
const int MOD = 1e9 + 7, MAXN = 200, MAXP = 300;
int T, N, M, K[MAXP + 5][MAXP + 5], col[MAXN + 5][MAXN + 5], unk[MAXN + 5][MAXN + 5], cnt;
char gar[MAXN + 5][MAXN + 5];
inline int qkpow ( int a, int b, const int p = MOD ) {
int ret = 1;
for ( ; b; a = 1ll * a * a % p, b >>= 1 ) ret = 1ll * ret * ( b & 1 ? a : 1 ) % p;
return ret;
}
inline int det ( const int n ) {
int ret = 1, swp = 1;
for ( int i = 2; i <= n; ++ i ) {
for ( int j = i; j <= n; ++ j ) {
if ( K[j][i] ) {
if ( i ^ j ) std::swap ( K[i], K[j] ), swp *= -1;
break;
}
}
if ( ! ( ret = 1ll * ret * K[i][i] % MOD ) ) return 0;
for ( int j = i + 1, inv = qkpow ( K[i][i], MOD - 2 ); j <= n; ++ j ) {
int d = 1ll * K[j][i] * inv % MOD;
for ( int k = i; k <= n; ++ k ) K[j][k] = ( K[j][k] - 1ll * K[i][k] * d % MOD + MOD ) % MOD;
}
}
return ( ret * swp + MOD ) % MOD;
}
inline bool findLoop ( const int x, const int y, const int cur ) {
if ( x < 1 || x > N || y < 1 || y > M || gar[x][y] == '.' ) return false;
if ( col[x][y] == cur ) return true;
if ( col[x][y] ) return false;
col[x][y] = cur;
if ( gar[x][y] == 'L' ) return findLoop ( x, y - 1, cur );
if ( gar[x][y] == 'R' ) return findLoop ( x, y + 1, cur );
if ( gar[x][y] == 'U' ) return findLoop ( x - 1, y, cur );
if ( gar[x][y] == 'D' ) return findLoop ( x + 1, y, cur );
return false;
}
inline int findUnknown ( const int x, const int y ) {
if ( x < 1 || x > N || y < 1 || y > M ) return 1;
if ( unk[x][y] ) return unk[x][y];
int& ret = unk[x][y];
if ( gar[x][y] == 'L' ) ret = findUnknown ( x, y - 1 );
if ( gar[x][y] == 'R' ) ret = findUnknown ( x, y + 1 );
if ( gar[x][y] == 'U' ) ret = findUnknown ( x - 1, y );
if ( gar[x][y] == 'D' ) ret = findUnknown ( x + 1, y );
return ret;
}
inline void add ( const int s, int t ) {
if ( ! t ) t = 1;
++ K[s][s], -- K[s][t];
if ( K[s][t] < 0 ) K[s][t] += MOD;
}
int main () {
for ( scanf ( "%d", &T ); T --; ) {
memset ( K, 0, sizeof K );
memset ( col, 0, sizeof col );
memset ( unk, 0, sizeof unk );
scanf ( "%d %d", &N, &M ), cnt = 1;
for ( int i = 1; i <= N; ++ i ) {
scanf ( "%s", gar[i] + 1 );
for ( int j = 1; j <= M; ++ j ) {
if ( gar[i][j] == '.' ) {
unk[i][j] = ++ cnt;
}
}
}
bool loop = false;
for ( int i = 1, cur = 1; i <= N && ! loop; ++ i ) {
for ( int j = 1; j <= M && ! loop; ++ j ) {
loop |= findLoop ( i, j, cur ++ );
}
}
if ( loop ) { puts ( "0" ); continue; }
for ( int i = 1; i <= N; ++ i ) {
for ( int j = 1; j <= M; ++ j ) {
findUnknown ( i, j );
}
}
for ( int i = 1; i <= N; ++ i ) {
for ( int j = 1; j <= M; ++ j ) {
if ( gar[i][j] == '.' ) {
add ( unk[i][j], unk[i][j - 1] );
add ( unk[i][j], unk[i][j + 1] );
add ( unk[i][j], unk[i - 1][j] );
add ( unk[i][j], unk[i + 1][j] );
}
}
}
printf ( "%d\n", det ( cnt ) );
}
return 0;
}
Solution -「Code+#2」「洛谷 P4033」白金元首与独舞的更多相关文章
- 「区间DP」「洛谷P1043」数字游戏
「洛谷P1043」数字游戏 日后再写 代码 /*#!/bin/sh dir=$GEDIT_CURRENT_DOCUMENT_DIR name=$GEDIT_CURRENT_DOCUMENT_NAME ...
- [LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞
[LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞 试题描述 到河北省 见斯大林 / 在月光下 你的背影 / 让我们一起跳舞吧 うそだよ~ 河北省怎么可能有 Stalin. ...
- Solution -「JSOI 2019」「洛谷 P5334」节日庆典
\(\mathscr{Description}\) Link. 给定字符串 \(S\),求 \(S\) 的每个前缀的最小表示法起始下标(若有多个,取最小的). \(|S|\le3\time ...
- Solution -「洛谷 P4372」Out of Sorts P
\(\mathcal{Description}\) OurOJ & 洛谷 P4372(几乎一致) 设计一个排序算法,设现在对 \(\{a_n\}\) 中 \([l,r]\) 内的元素排 ...
- Solution -「POI 2010」「洛谷 P3511」MOS-Bridges
\(\mathcal{Description}\) Link.(洛谷上这翻译真的一言难尽呐. 给定一个 \(n\) 个点 \(m\) 条边的无向图,一条边 \((u,v,a,b)\) 表示从 ...
- Solution -「APIO 2016」「洛谷 P3643」划艇
\(\mathcal{Description}\) Link & 双倍经验. 给定 \(n\) 个区间 \([a_i,b_i)\)(注意原题是闭区间,这里只为方便后文描述),求 \(\ ...
- 「洛谷P1516」 青蛙的约会
洛谷题号:P1516 出处:? 主要算法:数论 难度:4.4 思路分析: 典型的同余方程.由于是纬线,绕一圈是可以绕回来的,所以是可以取模的. 阅读题目,很容易得到同余方程$ x + tm ≡ y + ...
- 「BZOJ1038」「洛谷P2600」「ZJOI2008」瞭望塔 半平面交+贪心
题目链接 BZOJ/洛谷 题目描述 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安. 我们将H村抽象为一维的轮廓.如下图所示: 我们可以用一条山的上方 ...
- 「洛谷4197」「BZOJ3545」peak【线段树合并】
题目链接 [洛谷] [BZOJ]没有权限号嘤嘤嘤.题号:3545 题解 窝不会克鲁斯卡尔重构树怎么办??? 可以离线乱搞. 我们将所有的操作全都存下来. 为了解决小于等于\(x\)的操作,那么我们按照 ...
随机推荐
- Word2010格式化可爱的家乡
原文链接:https://www.toutiao.com/i6487795632349118990/ 准备样文 选中"可爱的家乡",选择"开始"选项卡,&quo ...
- vue中动画的使用
不要在router-view的外层使用动画!不要在router-view的外层使用动画!不要在router-view的外层使用动画! 重要的事情说三遍,在app.vue中自以为奇思妙想(实际是脑残)在 ...
- python多环境管理一(venv与virtualenv)
一.背景 我们经常会遇见这样的场景: 1.各个项目使用的python版本不相同 由于Python的解释器版本众多,各版本之间差异非常大.特别是python2和python3,互不兼容. 有些项目可能用 ...
- Texture+PBR两种工作流程
一.导入Texture 1.Inpspector TextureSize 2的n次幂,底层图形学需要,计算更快:不使用2的倍数,系统也会添加像素补全2n: 有最大尺寸限制8k,cubemap最高4k: ...
- JAVA-JDK1.7-ConCurrentHashMap-测试和验证
概述 上次记录了关于ConCurrentHashMap的原理以及源码,现在我对其进行有关功能的测试,下面就讲解一下我测试的内容和代码.这些测试主要针对JDK1.7版本. GET安全测试 上一篇写过ge ...
- 前端——JSON学习总结
学习网址: https://www.bilibili.com/video/BV1Pt411u7R3 什么是JSON?(以下有关概念内容为视频中学习文档相关内容,代码为本人学习时使用的有关代码) JSO ...
- Elasticsearch使用系列-ES简介和环境搭建
一.简介 Elasticsearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口.Elasticsearch是用Java语言开发的,并 ...
- Rust 连接 PostgreSQL 数据库
这次,我们使用 postgres 这个 crate 来连接和操作 PostgreSQL 数据库. 创建好项目后,在 cargo.toml 里添加 postgres 的依赖: 首先,导入相关的类型,并创 ...
- IDEA2020.1破解
IDEA2020.1破解 安装 下载idea idea官方下载地址:https://www.jetbrains.com/webstorm/download/other.html 下载破解插件 链接:h ...
- 不难懂-----git一套流程
001.初始化仓库,创建git仓库 git init 002.配置个人信息 git config --global user.name <名字> --------->:配置用户名 g ...