@description@

给定一个 H*W 黑白格图,保证黑格四连通。

定义分形如下:0 级分形是一个 1*1 的黑格;通过将第 k 级分形中的白格替换成 H*W 的全白,黑格替换成题目所给的 H*W 格图得到第 k + 1 级分形。

求第 K 级分形中黑格的四连通块个数。

原题传送门。

@solution@

最好先特判 K = 0 的情况。

如果存在一行最左最右都是黑格,且存在一列最上最下都是黑格,则黑格始终形成 1 个连通块。

如果不存在一行最左最右都是黑格,且不存在一列最上最下都是黑格,则连通块个数每次都呈指数增长。

如果记 cnt = 原图中黑格数量,则 ans = cnt^K。

否则,我们讨论存在一行最左最右都是黑格(另一种情况同理)。

记 g(i, j) 表示 j 个 i 级分形拼成一行,黑格形成的连通块数量,我们可以在 g(i, ...) 与 g(i - 1, ...) 之间建立转移关系。

讨论每一行是否全为黑格。如果不是,求每一行最左边往右延伸的最长连续黑格,最右边往左延伸的最长连续黑格,中间那些不与最左/右边有关的黑格连续段,利用这些信息进行转移。

具体怎么转移比较复杂,此处不提。

但是注意到 g 的第二维可能超大。不妨猜测 g(i, j) = k(i)*j + b(i),代入转移式发现成立。

于是就变成了 k(i), b(i) 与 k(i - 1), b(i - 1) 之间的转移。剩下的直接使用矩阵幂即可。

@accepted code@

#include <cstdio>
#include <algorithm>
using namespace std; typedef long long ll; const int MAXN = 1000;
const int MOD = int(1E9) + 7; inline int add(int x, int y) {return (x + y >= MOD ? x + y - MOD : x + y);}
inline int sub(int x, int y) {return (x - y < 0 ? x - y + MOD : x - y);}
inline int mul(int x, int y) {return 1LL * x * y % MOD;} int pow_mod(int b, ll p) {
int ret = 1;
for(ll i=p;i;i>>=1,b=mul(b,b))
if( i & 1 ) ret = mul(ret, b);
return ret;
} struct matrix{
int a[2][2];
friend matrix operator * (matrix A, matrix B) {
matrix C;
C.a[0][0] = add(mul(A.a[0][0], B.a[0][0]), mul(A.a[0][1], B.a[1][0]));
C.a[0][1] = add(mul(A.a[0][0], B.a[0][1]), mul(A.a[0][1], B.a[1][1]));
C.a[1][0] = add(mul(A.a[1][0], B.a[0][0]), mul(A.a[1][1], B.a[1][0]));
C.a[1][1] = add(mul(A.a[1][0], B.a[0][1]), mul(A.a[1][1], B.a[1][1]));
return C;
}
}; matrix mpow(matrix A, ll p) {
matrix ret; ret.a[0][0] = ret.a[1][1] = 1, ret.a[0][1] = ret.a[1][0] = 0;
for(ll i=p;i;i>>=1,A=A*A)
if( i & 1 ) ret = ret*A;
return ret;
} int a, b[MAXN + 5], c[MAXN + 5], d[MAXN + 5]; char s[MAXN + 5][MAXN + 5], t[MAXN + 5][MAXN + 5];
int main() {
int H, W; ll K; scanf("%d%d%lld", &H, &W, &K);
for(int i=0;i<H;i++)
scanf("%s", s[i]);
if( K == 0 ) {
puts("1");
return 0;
}
bool f1 = false, f2 = false;
for(int i=0;i<H;i++)
if( s[i][0] == '#' && s[i][W-1] == '#' ) {
f1 = true;
break;
}
for(int i=0;i<W;i++)
if( s[0][i] == '#' && s[H-1][i] == '#' ) {
f2 = true;
break;
}
if( f1 && f2 )
puts("1");
else if( (!f1) && (!f2) ) {
int cnt = 0;
for(int i=0;i<H;i++)
for(int j=0;j<W;j++)
cnt += (s[i][j] == '#');
printf("%d\n", pow_mod(cnt, K - 1));
}
else {
if( f2 ) {
for(int i=0;i<H;i++)
for(int j=0;j<W;j++)
t[j][i] = s[i][j];
swap(H, W);
for(int i=0;i<H;i++)
for(int j=0;j<W;j++)
s[i][j] = t[i][j];
}
for(int i=0;i<H;i++) {
bool flag = true;
for(int j=0;j<W;j++)
if( s[i][j] == '.' ) {
flag = false;
break;
}
if( flag ) a++;
else {
int l, r;
for(int j=0;j<W;j++)
if( s[i][j] == '.' ) {
l = j;
break;
}
for(int j=W-1;j>=0;j--)
if( s[i][j] == '.' ) {
r = j;
break;
}
int lst = l;
for(int j=l;j<=r;j++) {
if( s[i][j] == '.' ) {
if( lst != j ) b[j-lst]++;
lst = j + 1;
}
}
c[i] = l, d[i] = W - r - 1;
}
}
matrix A;
A.a[0][0] = mul(a, W), A.a[1][1] = a, A.a[0][1] = A.a[1][0] = 0;
for(int j=1;j<=W;j++)
A.a[0][0] = add(A.a[0][0], mul(b[j], j)), A.a[0][1] = add(A.a[0][1], b[j]);
for(int j=0;j<H;j++) {
if( c[j] ) A.a[1][0] = add(A.a[1][0], c[j]), A.a[1][1] = add(A.a[1][1], 1);
if( d[j] ) A.a[1][0] = add(A.a[1][0], d[j]), A.a[1][1] = add(A.a[1][1], 1);
if( c[j] + d[j] ) {
A.a[0][0] = add(A.a[0][0], c[j] + d[j]), A.a[0][1] = add(A.a[0][1], 1);
A.a[1][0] = sub(A.a[1][0], c[j] + d[j]), A.a[1][1] = sub(A.a[1][1], 1);
}
}
A = mpow(A, K - 1);
printf("%d\n", add(A.a[0][1], A.a[1][1]));
/*
int x = 0, y = 1;
for(int i=2;i<=K;i++) {
int x1 = mul(a, mul(x, W)), y1 = mul(a, y);
for(int j=1;j<=W;j++)
x1 = add(x1, mul(b[j], add(mul(x, j), y))); // x1 += b[j]*(x*j + y)
for(int j=0;j<H;j++) {
if( c[j] ) y1 = add(y1, add(mul(x, c[j]), y)); // y1 += c[j]*x + y
if( d[j] ) y1 = add(y1, add(mul(x, d[j]), y)); // y1 += d[j]*x + y
if( c[j] + d[j] ) {
x1 = add(x1, add(mul(x, c[j] + d[j]), y)); // x1 += (c[j]+d[j])*x + y
y1 = sub(y1, add(mul(x, c[j] + d[j]), y)); // y1 -= (c[j]+d[j])*x + y
}
}
x = x1, y = y1;
}
printf("%d\n", add(x, y));
*/
}
}

@details@

感觉分形题真有意思。

@atcoder - AGC003F@ Fraction of Fractal的更多相关文章

  1. [agc003F]Fraction of Fractal

    Description 传送门 Solution 本篇博客思路来自大佬的博客(侵删). 我们定义如果网格的第一行和最后一行的第i列都为黑色,则它是一个上下界接口.左右界接口定义同上. 如果上下界接口和 ...

  2. [AGC003F] Fraction of Fractal 矩阵快速幂

    Description ​ SnukeSnuke从他的母亲那里得到了生日礼物--一个网格.网格有HH行WW列.每个单元格都是黑色或白色.所有黑色单元格都是四联通的,也就是说,只做水平或垂直移动且只经过 ...

  3. [AGC003F] Fraction of Fractal(矩阵乘法)

    Description Snuke从他的母亲那里得到了生日礼物--一个网格.网格有H行W列.每个单元格都是黑色或白色.所有黑色单元格都是四联通的,也就是说,只做水平或垂直移动且只经过黑色单元格即可从任 ...

  4. AtCoder Grand Contest 003 F - Fraction of Fractal

    题目传送门:https://agc003.contest.atcoder.jp/tasks/agc003_f 题目大意: 给定一个\(H×W\)的黑白网格,保证黑格四连通且至少有一个黑格 定义分形如下 ...

  5. Atcoder Grand Contest 003 F - Fraction of Fractal(矩阵乘法)

    Atcoder 题面传送门 & 洛谷题面传送门 Yet another AGC F,然鹅这次就没能自己想出来了-- 首先需注意到题目中有一个条件叫做"黑格子组成的连通块是四联通的&q ...

  6. 【AGC003F】Fraction of Fractal

    Description ​ 原题链接 ​ Solution ​ 神题. ​ 定义一个上边界或下边界的格子为"上下接口",当且仅当上下边界该位置的格子都是黑色的. ​ "左 ...

  7. 题解-AtCoder-agc003F Fraction of Fractal(非矩阵快速幂解法)

    Problem AtCoder-agc003F 题意:给出\(n\)行\(m\)列的01矩阵,一开始所有 \(1\) 连通,称此为\(1\)级分形,定义\(i\)级分形为\(i-1\)级分形中每个标示 ...

  8. AtCoder Grand Contest 003

    AtCoder Grand Contest 003 A - Wanna go back home 翻译 告诉你一个人每天向哪个方向走,你可以自定义他每天走的距离,问它能否在最后一天结束之后回到起点. ...

  9. Atcoder/Topcoder 口胡记录

    Atcoder/Topcoder 理论 AC Atcoder的❌游戏示范 兴致勃勃地打开一场 AGC 看 A 题,先 WA 一发,然后花了一年时间 Fix. 看 B 题,啥玩意?这能求? 睡觉觉. e ...

随机推荐

  1. mysql小白系列_04 binlog(未完)

    mysql打开.查看.清理binlog 1.开启日志 log_bin=/var/lib/mysql/mysql-bin mysql> show variables like '%log_bin% ...

  2. java中的上下问解释以及ServletContext介绍使用

    摘抄的:所谓上下文,它是用来存储系统的一些初始化信息,例如在jboss中通过配置文件指定了数据源,那么在jboss启动的时候就把这个文件的相关信息上下文中,于是在我们使用这个数据源的时候,就需要先获得 ...

  3. hdu4757 可持续字典树

    Tree Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 102400/102400 K (Java/Others)Total Sub ...

  4. 【深度学习】PyTorch之Squeeze()和Unsqueeze()

    1. unsqueeze() 该函数用来增加某个维度.在PyTorch中维度是从0开始的. import torch a = torch.arange(0, 9) print(a) 结果: tenso ...

  5. 赛艇表演 51nod提高组模拟试题

    AC通道 题目描述 小明去某个地区观看赛艇比赛,这个地区共有n个城市和m条道路,每个城市都有赛艇比赛,在第i个 城市观看赛艇表演的价钱为ai, 去其他城市观看也需要支付赛艇表演的价格.任意两个城市之间 ...

  6. meavn项目由打包方式jar改为war报Cannot install Dynamic Web Module 2.5 facet. It is incompatibile with already installed facets: Utility Module. Please modify project configuration.处理方式

    找到  \项目名\.setting\文件夹下的   org.eclipse.wst.common.project.facet.core.xml  xml文件. 添加或修改 <installed ...

  7. ## H5 canvas画图白板踩坑

    最近接手了一个小型的H5,最主要的功能大概就是拍照上传和canvas画板了. 主要是记录一下自己菜到像傻子一样的技术. 1.canvas画板隔空打牛!画布越往上部分错位距离越小,越往下距离越大. 2. ...

  8. 舵机MX-64AR与MX-28AR驱动

    背景:硬件采用485通信,在tb上采购的无需收发控制的串口转RS485模块(485通信为半双工,一般情况需要控制收发模式).在使用该模块后,即可完全使用一个普通地串口来对485通信的舵机进行操作. 模 ...

  9. 网站的安全性对seo优化至关重要-智狐SEO顾问

    网站的安全性对seo优化至关重要    作者:智狐zhihuseo 从大的范围来看,网站安全性能也属于seo的范畴之一.域名被恶意泛解析就是网站安全性能低下的特征之一,如果网站域名被恶意泛解析,会直接 ...

  10. .net core 上传大文件

    using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Thr ...