Description

给个n*m的网格,里面一些格子一开始是白色(0),另一些是黑色(1),接下来网格可进行迭代,对于每次迭代,格子里颜色的变化遵循以下规则:若该格子有相邻格子的颜色与之相同,则颜色翻转;若该格子没有相邻格子的颜色与之相同,则颜色不变。有t次询问,每次询问都有i, j, p,表示问第i行第j列格子在第p次迭代后是什么颜色。

Solution

如果相邻的两个点有一样的颜色,就会在下一次操作中保持同样的颜色并开始变色

考虑求出 \(f[i][j]\) 表示第 \((i,j)\) 个点在多少次迭代后开始变色

如果一个格子一开始就能变色,设 \(f[i][j]=0\),否则设 \(f[i][j] = \infty\)

从开始就能变色的格子开始 BFS,当做边权为 \(1\) 的最短路问题 BFS 即可

处理询问时,如果一个格子的 \(f[i][j]=\infty\),则取其原始颜色;如果 \(time<f[i][j]\),也取其原始颜色;如果 \(time \ge f[i][j]\),则考虑 \(time-f[i][j]\) 的奇偶性,如果是奇数则与原始颜色相反,偶数则相同

#include <bits/stdc++.h>
using namespace std;


#define int long long
const int N = 1005;
const int inf = 2e18;

struct point {int x,y;};

int n,m,q,x,y,z,f[N][N],a[N][N];

signed main() {
    ios::sync_with_stdio(false);
    cin>>n>>m>>q;
    memset(f,0x3f,sizeof f);
    memset(a,0xff,sizeof a);
    for(int i=1;i<=n;i++) {
        string s;
        cin>>s;
        for(int j=1;j<=m;j++) a[i][j]=s[j-1]=='1';
    }
    queue <point> qu;
    for(int i=1;i<=n;i++) {
        for(int j=1;j<=m;j++) {
            if(a[i][j]==a[i-1][j] || a[i][j]==a[i+1][j] || a[i][j]==a[i][j-1] || a[i][j]==a[i][j+1]) {
                f[i][j]=0;
                qu.push({i,j});
            }
        }
    }
    while(qu.size()) {
        int i=qu.front().x,j=qu.front().y;
        qu.pop();
        int ni,nj;
        ni=i+1,nj=j;
        if(ni>=1 && nj>=1 && ni<=n && nj<=m) {
            if(f[ni][nj]>f[i][j]+1) {
                f[ni][nj]=f[i][j]+1;
                qu.push({ni,nj});
            }
        }
        ni=i-1,nj=j;
        if(ni>=1 && nj>=1 && ni<=n && nj<=m) {
            if(f[ni][nj]>f[i][j]+1) {
                f[ni][nj]=f[i][j]+1;
                qu.push({ni,nj});
            }
        }
        ni=i,nj=j-1;
        if(ni>=1 && nj>=1 && ni<=n && nj<=m) {
            if(f[ni][nj]>f[i][j]+1) {
                f[ni][nj]=f[i][j]+1;
                qu.push({ni,nj});
            }
        }
        ni=i,nj=j+1;
        if(ni>=1 && nj>=1 && ni<=n && nj<=m) {
            if(f[ni][nj]>f[i][j]+1) {
                f[ni][nj]=f[i][j]+1;
                qu.push({ni,nj});
            }
        }
    }
    for(int i=1;i<=q;i++) {
        cin>>x>>y>>z;
        if(z<f[x][y]) {
            cout<<a[x][y]<<endl;
        }
        else {
            cout<<(a[x][y]^((f[x][y]-z)&1))<<endl;
        }
    }
}

相关文章: