문제
세 마을의 좌표가 (x1, y1, z1), (x2, y2, z2), (x3, y3, z3)이라고 가정해보자. 이때, 세 마을을 친밀도는 아래와 같이 구할 수 있다.
친밀도 = d12 + d23 (dij = |xi - xj| + |yi - yj| + |zi - zj|)
마을이 주어졌을 때, 가장 작은 세 마을의 친밀도를 구하는 프로그램을 작성하시오.
입력
첫째 줄에 마을의 수 N (3 ≤ N ≤ 10,000)이 주어진다. 다음 N개 줄에는 마을의 위치 (x, y, z)가 주어진다. (-1000 ≤ x,y,z ≤ 1000)
출력
세 마을의 친밀도 중 가장 작은 값을 출력한다.
풀이
친밀도 공식에서 d2가 중복된다. 이를 이용하여 b에서 양쪽으로 이동하는 거리를 생각하여 가장 짧은 거리를 찾는다.
작은 거리를 저장하기 위해서 가장 작은 값, 그 다음으로 작은 값을 찾아야 한다.
배열을 여러개 만들어서 사용하지 않고 bigmin값과 smallmin값을 이용하여 가장 계속하여 bigm에 작은 값을 갱신하고
갱신 후, bigm과 smallm을 비교하여 항상 smallm을 bigm보다 작거나 같도록 유지한다.
실패기록
1. 메모리 초과
int [][] arr = new int[n][3], dist = new int[n][n];
a->b로 가는 모든 경로를 생성해서 a->b->c를 계산하려고함
메모리 초과 => (int 4 byte) * 10000 * 10000 = 400Mb
2. 시간 초과
for(int j=0;j<n;j++) {
if(i==j)
dist[j]=9999;
else {
for(int k=0;k<3;k++)
dist[j]=Math.abs(arr[i][k]-arr[j][k]);
}
}
Arrays.sort(dist);
ans=ans>dist[0]+dist[1]?dist[0]+dist[1]:ans;
}
for문을 2두번 돌리기에 이미 최악의 경우 10000*10000 => 1억번의 연산을 하며 추가적으로 Arrays.sort를 수행하기에 대략 O(n^4)를 가져가는 코드가 됨
코드
import java.io.*;
import java.util.*;
public class Main {
public static void main(String[] args) throws Exception{
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
int n=Integer.parseInt(br.readLine()), ans=Integer.MAX_VALUE;
int [][] arr = new int[n+1][3];
for(int i=1;i<=n;i++) {
StringTokenizer st = new StringTokenizer(br.readLine());
for(int j=0;j<3;j++) // 각각의 좌표 3점을 베열형태로 입력받음
arr[i][j]=Integer.parseInt(st.nextToken());
}
// a-> b, b-> c의 거리를 연산하기에 b는 중복됨
// b를 기준으로 삼고 a의 거리 c의 거리를 구해서, 가장 작은 2개의 값을 더하면 b->x, b->y로의 거리를 구하기에 다르게 생각하면 x->b->y와 같은 결과를 출력
for(int i=1;i<=n;i++) { // b를 i로 생각
int bigm=9999, smallm=9999; // 시간 초과 오류를 해결하기 위해서 최소값1, 최소값2를 만들어준다.
for(int j=1;j<=n;j++) { // b에서 다른 위치로 가는 경로의 길이를 구하기 위한 탐색문
int dist=0;
if(i!=j) { // b->b로 가는 경로는 생길 수 없음
for(int k=0;k<3;k++) // 좌표가 x, y, z 3개 이기에 저장형태를 반복으로 계산
dist+=Math.abs(arr[i][k]-arr[j][k]);
if(dist<bigm) { // 최소값중에서 작은 최소값 큰 최소값 두개를 만들고 큰 최소값과 비교해서 현재 계산된 최소값이 더 작다면 교환
bigm=dist;
if(smallm>bigm) { // 갱신된 최소값이 작은 최소값보다 작은지 비교하고, 작은 최소값에 더 작은 최소값을 더하기 위해 swap연산을 수행
int temp=smallm;
smallm=bigm;
bigm=temp;
}
}
}
}
ans=Math.min(smallm+bigm, ans); // 현재까지의 최소값보다 더 작다면 그 작은 값이 답이 됨
if(ans==2)
break;
}
System.out.println(ans);
}
}
'Algorithm > Baekjoon' 카테고리의 다른 글
Baekjoon 9550 아이들은 사탕을 좋아해 JAVA (0) | 2022.06.09 |
---|---|
Baekjoon 5355 화성 수학 JAVA (0) | 2022.06.09 |
Baekjoon 4101 크냐? JAVA (0) | 2022.06.09 |
Baekjoon 3613 Java vs C++ JAVA (0) | 2022.06.09 |
Baekjoon 21966 (중략) JAVA (0) | 2022.06.09 |
댓글