티스토리 뷰
반응형
✏️ 문제 링크
https://www.acmicpc.net/problem/2146
✏️ 문제 설명
✏️ 문제 풀이
먼저 주어진 좌표들을 이용해서 bfs를 돌려서 각 섬의 영역들을 정해줍니다.
그러면 예제입력의 경우에는 총 섬이 3덩어리로 나눠지게 됩니다.
그 다음, 각 섬들의 영역에서 bfs로 돌려서 다른 섬으로 갈 수 있는 다리들의 최솟값을 구해줍니다.
자세한 bfs조건들은 아래 코드를 봐주세요
✏️ 문제 코드
#include<iostream>
#include<queue>
#include<vector>
using namespace std;
typedef pair<int, int>cur;
int area[101][101];
int dist[101][101];
bool visit[101][101];
int dx[4] = { 0,1,0,-1 };
int dy[4] = { 1,0,-1,0 };
int N,now;
void bfs(int i, int j);
int main() {
ios::sync_with_stdio(false);
cin.tie(NULL);
cout.tie(NULL);
cin >> N;
for (int i = 0; i < N; i++) {
for (int j = 0; j < N; j++) {
cin >> area[i][j];
dist[i][j] = -1;
}
}
for (int i = 0; i < N; i++) {
for (int j = 0; j < N; j++) {
if (visit[i][j] == false && area[i][j] == 1) {
now++;
bfs(i, j);
}
}
}
int min_val = 987654321;
for (int i = 0; i < N; i++) {
for (int j = 0; j < N; j++) {
if (area[i][j] == 0)
continue;
for (int ll = 0; ll < N; ll++) {
for (int jj = 0; jj < N; jj++)
dist[ll][jj] = -1;
}
queue<cur>q;
q.push({ i,j });
dist[i][j] = 0;
int now_m = area[i][j];
while (!q.empty()) {
cur tmp = q.front();
int x1 = tmp.first;
int y1 = tmp.second;
q.pop();
for (int k = 0; k < 4; k++) {
int mx = x1 + dx[k];
int my = y1 + dy[k];
if (mx < 0 || my < 0 || mx >= N || my >= N)
continue;
if (dist[mx][my] != -1)
continue;
if (area[mx][my] == now_m)
continue;
if (area[mx][my] != 0)
min_val = min(min_val, dist[x1][y1]);
else {
dist[mx][my] = dist[x1][y1] + 1;
q.push({ mx,my });
}
}
}
}
}
cout << min_val;
}
void bfs(int i, int j) {
queue<cur>q;
q.push({ i,j });
visit[i][j] = true;
area[i][j] = now;
while (!q.empty()) {
cur tmp = q.front();
int x1 = tmp.first;
int y1 = tmp.second;
q.pop();
for (int i = 0; i < 4; i++) {
int mx = x1 + dx[i];
int my= y1 + dy[i];
if (mx < 0 || my < 0 || mx >= N || my >= N)
continue;
if (area[mx][my] == 0 || visit[mx][my])
continue;
area[mx][my] = now;
q.push({ mx,my });
visit[mx][my] = true;
}
}
}
✏️ 문제 링크
https://www.acmicpc.net/problem/17472
✏️ 문제 설명
✏️ 문제 풀이
기본적인 문제 풀이 과정은 위에서 언급했던 2146번과 동일합니다. 그러나 이 문제의 경우에는 최소 신장 트리 알고리즘도 같이 이용해야 합니다. 위의 문제에서는 어느 섬에서든 상관없이 다리길이의 최솟값을 구했다고 하면, 이 문제에서는 각 섬에서 자기 자신을 제외한 모든 다리길이를 저장해야 합니다.
다리 길이를 저장할 때는, (출발섬, 도착섬, 다리길이를 모두 저장)해야 하고 다리길이를 기준으로 오름차순 정렬을 해줍니다. 그런 다음 최소 신장 트리 알고리즘을 진행해 주시면 됩니다.
✏️ 문제 코드
#include <iostream>
#include <queue>
using namespace std;
void munion(int a, int b);
int find(int a);
void BFS(int i, int j);
static int dr[] = { -1, 0, 1, 0 };
static int dc[] = { 0, 1, 0, -1 };
static int map[101][101];
static bool visited[101][101] = { false, };
static vector<int> parent;
static int N, M, sNum;
static vector < vector <pair<int, int>> > sumlist;
static vector <pair<int, int>> mlist;
typedef struct Edge // 에지정보 struct 생성, 가중치 값 기준 오름차순 정렬로 설정
{
int s, e, v;
bool operator> (const Edge& temp) const {
return v > temp.v;
}
}Edge;
static priority_queue<Edge, vector<Edge>, greater<Edge>> pq; // 오름차순 정렬
int main()
{
ios::sync_with_stdio(false);
cin.tie(NULL);
cout.tie(NULL);
cin >> N >> M;
for (int i = 0; i < N; i++) {
for (int j = 0; j < M; j++) {
cin >> map[i][j]; // 맵 정보 저장
}
}
sNum = 1;
for (int i = 0; i < N; i++) { // 각 자리에서 BFS 탐색을 이용하여 섬들을 분리하여 줍니다.
for (int j = 0; j < M; j++) {
if (map[i][j] != 0 && visited[i][j] != true) {
BFS(i, j);
sNum++;
sumlist.push_back(mlist);
}
}
}
for (int i = 0; i < sumlist.size(); i++) { // 섬의 각 지점에서 만들 수 있는 모든 간선을 저장
vector<pair<int, int>> now = sumlist[i];
for (int j = 0; j < now.size(); j++) {
int r = now[j].first;
int c = now[j].second;
int now_S = map[r][c];
for (int d = 0; d < 4; d++) { // 4방향 검색
int tempR = dr[d];
int tempC = dc[d];
int blenght = 0;
while (r + tempR >= 0 && r + tempR < N && c + tempC >= 0 && c + tempC < M) {
if (map[r + tempR][c + tempC] == now_S) // 같은 섬이면 간선을 만들 수 없음
break;
else if (map[r + tempR][c + tempC] != 0) { //같은 섬이 아니고 바다가 아니면
if (blenght > 1) // 다른 섬 -> 길이가 1이상일때 간선으로 더해줍니다.
pq.push(Edge{ now_S, map[r + tempR][c + tempC], blenght });
break;
}
else //바다이면 다리의 길이를 연장하여 줍니다.
blenght++;
if (tempR < 0)tempR--;
else if (tempR > 0)tempR++;
else if (tempC < 0)tempC--;
else if (tempC > 0)tempC++;
}
}
}
}
parent.resize(sNum);
for (int i = 0; i < parent.size(); i++) {
parent[i] = i;
}
int useEdge = 0;
int result = 0;
while (!pq.empty()) { // 최소 신장 트리 알고리즘을 수행하여 줍니다.
Edge now = pq.top();
pq.pop();
if (find(now.s) != find(now.e)) // 같은 부모가 아니라면 -> 연결 가능
{
munion(now.s, now.e);
result = result + now.v;
useEdge++;
}
}
// 배열에서 쉬운 index 처리를 위해 sNum을 1부터 시작하였으므로 현재 sNum의 값이 섬의 개수보다 1 많은 상태임 때문에 1작은 수가 아닌 2작은 수와 사용 에지를 비교하여 줍니다.
if (useEdge == sNum - 2) {
cout << result << "\n";
}
else {
cout << -1 << "\n";
}
}
void munion(int a, int b) { // union 연산 : 대표 노드끼리 연결하여 줌
a = find(a);
b = find(b);
if (a != b) {
parent[b] = a;
}
}
int find(int a) { // find 연산
if (a == parent[a])
return a;
else
return parent[a] = find(parent[a]); // 재귀함수의 형태로 구현 -> 경로 압축 부분
}
void BFS(int i, int j) { // BFS를 통하여 연결된 섬을 찾아줍니다.
queue<pair<int, int>> myqueue;
mlist.clear();
myqueue.push(make_pair(i, j));
mlist.push_back(make_pair(i, j));
visited[i][j] = true;
map[i][j] = sNum;
while (!myqueue.empty()) {
int r = myqueue.front().first;
int c = myqueue.front().second;
myqueue.pop();
for (int d = 0; d < 4; d++) { //4방향 검색
int tempR = dr[d];
int tempC = dc[d];
while (r + tempR >= 0 && r + tempR < N && c + tempC >= 0 && c + tempC < M) {
//현재 방문한 적이 없고 바다가 아니면 같은 섬으로 취급
if (visited[r + tempR][c + tempC] == false && map[r + tempR][c + tempC] != 0) {
int now_i = r + tempR;
int now_j = c + tempC;
map[now_i][now_j] = sNum;
visited[now_i][now_j] = true;
mlist.push_back(make_pair(now_i, now_j));
myqueue.push(make_pair(now_i, now_j));
}
else break;
if (tempR < 0)tempR--;
else if (tempR > 0)tempR++;
else if (tempC < 0)tempC--;
else if (tempC > 0)tempC++;
}
}
}
}
✏️ 참조
https://pooreumjung.tistory.com/302
https://pooreumjung.tistory.com/289
반응형
'Algorithm > BOJ' 카테고리의 다른 글
[C/C++] 백준 18436번 - 수열과 쿼리 37 (0) | 2024.04.08 |
---|---|
[C/C++] 백준 2610번 - 회의준비 (0) | 2024.04.08 |
[C/C++] 백준 LCS문제 모음 / 백준 9251번, 백준 9252번, 백준 1958번 (2) | 2024.04.07 |
[C/C++] 백준 외판원 순회 문제 / 백준 2098번, 백준 10971번, 백준 16991번 (0) | 2024.04.06 |
[C/C++] 백준 11780번 - 플로이드 2 (0) | 2024.04.06 |
반응형
공지사항
최근에 올라온 글
최근에 달린 댓글
- Total
- Today
- Yesterday
링크
TAG
- 백준 풀이
- 이분 매칭
- html
- 투 포인터
- 백준
- DFS
- Do it!
- 유니온 파인드
- HTML5
- 자바
- DP
- 유클리드 호제법
- C++
- 에라토스테네스의 체
- 반복문
- 카운팅 정렬
- 스프링 부트 crud 게시판 구현
- 알고리즘
- js
- 자바스크립트
- c++ string
- 알고리즘 공부
- CSS
- C++ Stack
- 스택
- 세그먼트 트리
- 우선순위 큐
- java
- BFS
- 자료구조
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
글 보관함