天天看点

Aizu 0558 Cheeseチーズ ()

チーズ ()

問題

今年も JOI 町のチーズ工場がチーズの生産を始め,ねずみが巣から顔を出した.JOI 町は東西南北に区画整理されていて,各区画は巣,チーズ工場,障害物,空き地のいずれかである.ねずみは巣から出発して全てのチーズ工場を訪れチーズを 1 個ずつ食べる.

この町には,N 個のチーズ工場があり,どの工場も1種類のチーズだけを生産している.チーズの硬さは工場によって異なっており,硬さ 1 から N までのチーズを生産するチーズ工場がちょうど 1 つずつある.

ねずみの最初の体力は 1 であり,チーズを 1 個食べるごとに体力が 1 増える.ただし,ねずみは自分の体力よりも硬いチーズを食べることはできない.

ねずみは,東西南北に隣り合う区画に 1 分で移動することができるが,障害物の区画には入ることができない.チーズ工場をチーズを食べずに通り過ぎることもできる.すべてのチーズを食べ終えるまでにかかる最短時間を求めるプログラムを書け.ただし,ねずみがチーズを食べるのにかかる時間は無視できる.

入力

入力は H+1 行ある.1 行目には 3 つの整数 H,W,N (1 ≤ H ≤ 1000,1 ≤ W ≤ 1000,1 ≤ N ≤ 9) がこの順に空白で区切られて書かれている.2 行目から H+1 行目までの各行には,'S','1', '2', ..., '9','X','.' からなる W 文字の文字列が書かれており,各々が各区画の状態を表している.北から i 番目,西から j 番目の区画を (i,j) と記述することにすると (1 ≤ i ≤ H, 1 ≤ j ≤ W),第 i+1 行目の j 番目の文字は,区画 (i,j) が巣である場合は 'S' となり,障害物である場合は 'X' となり,空き地である場合は '.' となり,硬さ 1, 2, ..., 9 のチーズを生産する工場である場合はそれぞれ '1', '2', ..., '9' となる.入力には巣と硬さ 1, 2, ..., N のチーズを生産する工場がそれぞれ 1 つずつある.他のマスは障害物または空き地であることが保証されている.ねずみは全てのチーズを食べられることが保証されている.

出力

すべてのチーズを食べ終えるまでにかかる最短時間(分)を表す整数を 1 行で出力せよ.

入出力例

入力例 1

3 3 1
S..
...
..1
      

出力例 1

4
      

入力例 2

4 5 2
.X..1
....X
.XX.S
.2.X.
      

出力例 2

12
      

入力例 3

10 10 9
.X...X.S.X
6..5X..X1X
...XXXX..X
X..9X...X.
8.X2X..X3X
...XX.X4..
XX....7X..
X..X..XX..
X...X.XX..
..X.......
      

出力例 3

91
      

問題文と自動審判に使われるデータは、情報オリンピック日本委員会が作成し公開している問題文と採点用テストデータです。

题意:

#include <cstdio>
#include <queue>
#include <cstring>

using namespace std;

const int M=1000+5;
int m, n, w;

char mapp[M][M];
bool v[M][M];
int xx[10];
int yy[10];

struct node
{
    int x;
    int y;
    int s;
};


int dx[4]={0,0,1,-1};
int dy[4]={1,-1,0,0};

int bfs(int x,int y,node nd)
{
    queue<node> q;
    node t;
    t.x=x;
    t.y=y;
    t.s=0;
    q.push(t);
    while (!q.empty())
    {
        node t=q.front();
        q.pop();

        if(t.x==nd.x&&t.y==nd.y)
        {
            return t.s;
        }

        for(int i=0;i<4;i++)
        {
            int nx=t.x+dx[i];
            int ny=t.y+dy[i];
            if(1<=nx&&nx<=m&&1<=ny&&ny<=n&&!v[nx][ny])
            {
                if(mapp[nx][ny]!='X')
                {
                    node tt;
                    tt.x=nx;
                    tt.y=ny;
                    tt.s=t.s+1;
                    v[nx][ny]=1;
                    q.push(tt);
                }
            }
        }



    }
    return 0;


}

int main() {
    scanf("%d %d %d", &m, &n, &w);
    for(int i=1;i<=m;i++)
    {
        getchar();
        for(int j=1;j<=n;j++)
        {
            scanf("%c",&mapp[i][j]);
            if('1'<=mapp[i][j]&&mapp[i][j]<='9')
            {
                xx[mapp[i][j]-'0']=i;
                yy[mapp[i][j]-'0']=j;
            }
            else if(mapp[i][j]=='S')
            {
                xx[0]=i;
                yy[0]=j;
            }
        }
    }



    int step=0;
    for(int k=0;k<w;k++)
    {
        memset(v,0, sizeof(v));
        node nd;
        nd.x=xx[k+1];
        nd.y=yy[k+1];
        nd.s=0;
        v[xx[k]][yy[k]]=1;
        step+=bfs(xx[k],yy[k],nd);
    }
    printf("%d\n",step);



    return 0;
}
           

从S到1到2到3到...的最短距离,注意shun