ACM-ICPC 2018 南京赛区现场赛 K. Kangaroo Puzzle (思维+构造)

题目链接:https://codeforc.es/gym/101981/attachments

题意:在 n * m 的平面上有若干个袋鼠和墙(1为袋鼠,0为墙),每次可以把所有袋鼠整体往一个方向移动一步(不能走出边界和不能走到墙),为在不超过5e4步的情况下能否把全部袋鼠聚集在同一个位置。

题解:先预处理每个袋鼠到其他袋鼠的初始方向,然后每次选两个不同的袋鼠,其中一个向另一个逼近,直到聚集在一起,然后重复该操作。因为n,m <= 20,所以最多只有400个袋鼠,而每两个不同的袋鼠逼近的过程中最多走 80 步,故操作次数上界是 3.2e4。

 

  1 #include <bits/stdc++.h>
  2 #define sd(a) scanf("%d",&a)
  3 #define sld(a) scanf("%lld",&a)
  4 #define mst(a,b) memset(a,b,sizeof a)
  5 #define mp make_pair
  6 #define all(a) a.begin(),a.end()
  7 typedef long long ll;
  8 #define pb push_back
  9 using namespace std;
 10 typedef pair<int, int> pii;
 11 const int maxn = 2e6 + 10;
 12 const int maxm = 1e6 +  10;
 13 const int inf = 0x3f3f3f3f;
 14 const ll mod = 1e9 + 7;
 15 const double eps = 1e-9;
 16 char s[33][33];
 17 int to[33][33][33][33];
 18 bool vis[33][33];
 19 vector<pii>v;
 20 int n, m;
 21 
 22 bool judge(int x, int y) {
 23     return 1 <= x && x <= n && 1 <= y && y <= m && s[x][y] == '1';
 24 }
 25 
 26 int dx[] = {1, 0, -1, 0}, dy[] = {0, 1, 0, -1};
 27 void pre(int x, int y) {
 28     mst(vis, 0);
 29     vis[x][y] = 1;
 30     vector<pii>v;
 31     queue<pii>q;
 32     for(int i = 0; i < 4; ++i) {
 33         v.clear();
 34         int nx = x + dx[i], ny = y + dy[i];
 35         if(!judge(nx, ny))
 36             continue;
 37         q.push(mp(nx, ny));
 38         for(; !q.empty();) {
 39             pii p = q.front();
 40             q.pop();
 41             int ux = p.first;
 42             int uy = p.second;
 43             if(vis[ux][uy])
 44                 continue;
 45             vis[ux][uy] = 1;
 46             v.pb(p);
 47             for(int j = 0; j < 4; ++j) {
 48                 int qx = dx[j] + ux, qy = dy[j] + uy;
 49                 if(judge(qx, qy) && !vis[qx][qy])
 50                     q.push(mp(qx, qy));
 51             }
 52         }
 53         int sz = v.size();
 54         for(int j = 0; j < sz; ++j) {
 55             int qx = v[j].first;
 56             int qy = v[j].second;
 57             to[x][y][qx][qy] = i;
 58         }
 59     }
 60 }
 61 
 62 int main() {
 63 #ifdef LOCAL
 64     freopen("in", "r", stdin);
 65 #endif // local
 66 //    int n,m;
 67     sd(n), sd(m);
 68     for(int i = 1; i <= n; ++i)
 69         scanf("%s", s[i] + 1);
 70     for(int i = 1; i <= n; ++i)
 71         for(int j = 1; j <= m; ++j) {
 72             if(s[i][j] - '0')
 73                 v.pb(mp(i, j));
 74             pre(i, j);
 75         }
 76     vector<int>q;
 77     int sz = v.size();
 78     char ch[] = {'D', 'R', 'U', 'L'};
 79     for(;;) {
 80         int idx1 = 0, idx2 = -1;
 81         for(int i = 1; i < sz; ++i) {
 82             if(v[i] != v[idx1]) {
 83                 idx2 = i;
 84                 break;
 85             }
 86         }
 87         if(idx2 == -1)
 88             break;
 89         for(; v[idx1] != v[idx2];) {
 90             int x1 = v[idx1].first, x2 = v[idx2].first;
 91             int y1 = v[idx1].second, y2 = v[idx2].second;
 92             int id = to[x1][y1][x2][y2];
 93             q.pb(id);
 94             for(int i = 0; i < sz; ++i) {
 95                 if(judge(v[i].first + dx[id], v[i].second + dy[id]))
 96                     v[i].first += dx[id], v[i].second += dy[id];
 97             }
 98         }
 99     }
100     for(int i = 0, z = q.size(); i < z; ++i) {
101         int id = q[i];
102         putchar(ch[id]);
103     }
104     return 0;
105 }

 

posted on 2018-11-19 19:13  scau_lok  阅读(1362)  评论(0编辑  收藏  举报

导航