본문 바로가기

BOJ/C++

미네랄

*클러스터들을 하나의 영역으로 보고 번호를 라벨링하여 구분했음 

*클러스터끼리 인접해있다면 내려가지 않음 

*내려가는 칸의 수는 min(바닥과의 높이 차, 다른 클러스터와의 차이)

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
#include <iostream>
#include <vector>
#include <queue>
#include <cstring>
 
using namespace std;
 
int r, c, n;
char map[101][101];
typedef pair<intint> pp;
typedef pair<pp, pp> pppp;
 
int visited[101][101];
int dx[4= {-1100}, dy[4= {00-11};
 
void print()
{
    for (int i = 0; i < r; i++)
    {
        for (int j = 0; j < c; j++)
        {
            cout << map[i][j];
        }
        cout << '\n';
    }
}
void drawCluster(int label, int cnt)
{
 
    while (cnt--)
    {
        for (int i = r - 1; i >= 0; i--)
        {
            for (int j = c - 1; j >= 0; j--)
            {
                if (visited[i][j] != label)
                {
                    continue//해당 클러스터가 아님
                }
                else
                {
                    map[i + 1][j] = map[i][j];
                    map[i][j] = '.';
                    visited[i + 1][j] = visited[i][j];
                    visited[i][j] = 0;
                }
            }
        }
    }
}
void checkCluster(int label)
{
    //클러스터들을 끌어내려준다. (내려올 수 있는 것들만)
    for (int i = 1; i <= label; i++)
    {
        //해당 클러스터 넘버에 맞는 것들이 공중에 얼마나 떠있는지 확인
        //동시에 두 개의 클러스터가 내려올 확률은 없기 때문에 하나만 찾으면 된다.
        bool flag = false;
        int idxx = 101;
        for (int k = 0; k < c; k++)
        {
            for (int j = r - 1; j >= 0; j--)
            {
                if (visited[j][k] != i)
                    continue//찾는 클러스터가 아님
                else
                {
                    //바닥과 마주해있거나, 다른 클러스터와 붙어있음
                    if (j == r - 1 || (visited[j + 1][k] != i && visited[j + 1][k]))
                    {
                        flag = true;
                        break;
                    }
                    idxx = min(idxx, r - (j+1));
 
                    for (int z = j + 1; z < r; z++)
                    {
                        if (visited[z][k] != i && visited[z][k])
                        {
                            idxx = min(idxx, z - j - 1);
                            break;
                        }
                    }
                }
            }
            if (flag)
                break;
        }
        if (!flag)
        {
            //내려올게 있음
            drawCluster(i, idxx);
        }
        
    }
}
void dfs(int x, int y, int label)
{
    if (visited[x][y])
        return;
    visited[x][y] = label;
 
    for (int i = 0; i < 4; i++)
    {
        int nx = x + dx[i];
        int ny = y + dy[i];
 
        if (nx < 0 || nx >= r || ny < 0 || ny >= c || visited[nx][ny] || map[nx][ny] == '.')
            continue;
 
        dfs(nx, ny, label);
    }
}
 
int numberingCluster()
{
    int label = 0;
 
    for (int i = r - 1; i >= 0; i--)
    {
        for (int j = c - 1; j >= 0; j--)
        {
            if (!visited[i][j] && map[i][j] == 'x')
            {
 
                ++label;
                dfs(i, j, label);
            }
        }
    }
    return label;
}
void throwBar(int h, int dir)
{
    queue<pp> q;
 
    if (dir % 2)
    {
        //오른쪽에서 왼쪽으로
        for (int i = c - 1; i >= 0; i--)
        {
            if (map[h][i] == 'x')
            {
                //미네랄이 있으면
                //클러스터가 끊기고 주변의 클러스터가 밑으로 끌어내려질 수 있는지 확인하낟.
                map[h][i] = '.';
                return;
            }
        }
    }
    else
    {
        for (int i = 0; i < c; i++)
        {
            if (map[h][i] == 'x')
            {
                //미네랄이 있으면
                //클러스터가 끊기고 주변의 클러스터가 밑으로 끌어내려질 수 있는지 확인하낟.
                map[h][i] = '.';
 
                return;
            }
        }
    }
}
 
int main()
{
    ios_base::sync_with_stdio(false);
    cin.tie(0);
 
    cin >> r >> c;
 
    for (int i = 0; i < r; i++)
    {
        for (int j = 0; j < c; j++)
        {
            cin >> map[i][j];
        }
    }
 
    cin >> n;
    for (int i = 0; i < n; i++)
    {
        int tmp;
        cin >> tmp;
 
        memset(visited, 0sizeof(visited));
        throwBar(r - tmp, i);
        int label = numberingCluster();
 
        checkCluster(label);
    }
    print();
 
    return 0;
}
http://colorscripter.com/info#e" target="_blank" style="color:#4f4f4ftext-decoration:none">Colored by Color Scripter

 

도움이 된 TC

10 20
..xxxxxxxxxxxxxxx... 
..x.............x... 
..x.............x... 
..x.............x... 
..x.............x... 
..x.............x... 
..x.............x... 
..xxxxxx........x... 
................x... 
x...............x... 
2
1 7

'BOJ > C++' 카테고리의 다른 글

새로운 게임2  (0) 2020.02.25
  (0) 2020.02.24
영화수집  (0) 2020.02.22
스도쿠  (0) 2020.02.21
빵집  (0) 2020.02.21