CH2601 电路维修

描述

Ha'nyu是来自异世界的魔女,她在漫无目的地四处漂流的时候,遇到了善良的少女Rika,从而被收留在地球上。Rika的家里有一辆飞行车。有一天飞行车的电路板突然出现了故障,导致无法启动。
电路板的整体结构是一个R行C列的网格(R,C≤500),如右图所示。每个格点都是电线的接点,每个格子都包含一个电子元件。电子元件的主要部分是一个可旋转的、连接一条对角线上的两个接点的短电缆。在旋转之后,它就可以连接另一条对角线的两个接点。电路板左上角的接点接入直流电源,右下角的接点接入飞行车的发动装置。

Ha'nyu发现因为某些元件的方向不小心发生了改变,电路板可能处于断路的状态。她准备通过计算,旋转最少数量的元件,使电源与发动装置通过若干条短缆相连。不过,电路的规模实在是太大了,Ha'nyu并不擅长编程,希望你能够帮她解决这个问题。

输入格式

输入文件包含多组测试数据。第一行包含一个整数T 表示测试数据的数目。
对于每组测试数据,第一行包含正整数R 和C,表示电路板的行数和列数。
之后R 行,每行C 个字符,字符是"/"和""中的一个,表示标准件的方向。

输出格式

对于每组测试数据,在单独的一行输出一个正整数,表示所需的缩小旋转次数。
如果无论怎样都不能使得电源和发动机之间连通,输出NO SOLUTION。

样例输入

1
3 5
\\/\\
\\///
/\\\\

样例输出

1

数据范围与约定

  • 对于40% 的数据,R,C≤5。
    对于100% 的数据,R,C≤500,T≤5。

样例解释

样例的输入对应于题目描述中的情况。
只需要按照下面的方式旋转标准件,就可以使得电源和发动机之间连通。

来源

杜宇飞,石家庄二中【Nescafé 5】杯NOIP模拟赛


思路

我们在格点建边,构成一个无向图。如果不用旋转就能到达,边权为0,否则为1。

然后广搜~

but 纯粹的广搜(或者SPFA?)是会炸的。。。

我得了70分。。。T了3个点。。。

我们可以考虑使用双端队列BFS

即如果边权为0,直接放到队首,如果边权为1,就和普通BFS一样放到队尾。

有时会出现这样一种情况:

最优值相同的点A,B都能更新C,A在队列中的位置在B前面。AC=1 BC=0

这样会导致最优值出现错误。所以我们还要判断能不能更新已有的点。

像上述这种情况,A更新C后,C‘会到队尾,B更新C后,C到队首,比之前的C'更快更新,C'不能再更新其他点,也就不会影响结果。

代码

#include<bits/stdc++.h>
using namespace std;
#define open(s) freopen( s".in", "r", stdin ), freopen( s".out", "w", stdout )
#define MAXN 505

int T;
int R, C;
char s[MAXN][MAXN];
int dis[MAXN][MAXN];
deque<int> Qx, Qy;
int x, y, tx, ty, t1, t2; char t;
int dir[4][2] = { 1, 1, -1, -1, 1, -1, -1, 1 };
int wh[4][2] = { 1, 1, 0, 0, 1, 0, 0, 1 };

int main(){
    scanf( "%d", &T );
    while( T-- ){
        scanf( "%d%d", &R, &C );
        for ( int i = 1; i <= R; ++i ) scanf( "%s", s[i] + 1 );
        memset( dis, 0x7f, sizeof dis );
        dis[0][0] = 0;
        Qx.push_back(0); Qy.push_back(0);
        while( !Qx.empty() ){
            x = Qx.front(); y = Qy.front(); Qx.pop_front(); Qy.pop_front();
            for ( int i = 0; i < 4; ++i ){
                tx = x + dir[i][0]; ty = y + dir[i][1];
                t1 = x + wh[i][0]; t2 = y + wh[i][1];
                t = i <= 1 ? '\\' : '/';
                if ( tx >= 0 && ty >= 0 && tx <= R && ty <= C && dis[tx][ty] > dis[x][y] + ( s[t1][t2] != t )  ){
                    dis[tx][ty] = dis[x][y] + ( s[t1][t2] != t );
                    if ( s[t1][t2] != t ) Qx.push_back(tx), Qy.push_back(ty);
                    else Qx.push_front(tx), Qy.push_front(ty);
                }
            }
        }
        if ( dis[R][C] >= 0x3f3f3f3f ) printf( "NO SOLUTION\n" );
        else printf( "%d\n", dis[R][C] );
    }
    return 0;
}