ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • [SW Expert Academy] 1251. 하나로 ( 인접리스트 & 인접행렬 )
    Algorithm/Source Code 2022. 8. 25. 22:16
    반응형

    adjList.java

    public class SWEA1251_하나로_프림인접리스트 {
    	// 정점 중심
    	static class Node implements Comparable<Node>{
    		int vertex; // 노드
    		long weight; // 가중치
    		public Node(int vertex, long weight) {
    			super();
    			this.vertex = vertex;
    			this.weight = weight;
    		}
    		@Override
    		public int compareTo(Node o) {
    			return Long.compare(this.weight, o.weight);
    		}
    	}
    
    	public static void main(String[] args) throws IOException { // 0825 오전 라이브 강의 인접리스트로 prim 구현 참고 
    		BufferedReader bf = new BufferedReader(new InputStreamReader(System.in));
    		int T = Integer.parseInt(bf.readLine()); // 테스트 케이스 개수
    		for(int tc=1;tc<=T;tc++) {
    			int N = Integer.parseInt(bf.readLine()); // 정점의 수
    			boolean[] visited = new boolean[N]; // 신장트리에 포함 여부
    			
    			long[][] xy = new long[N][2]; // x y 입력받기
    			String[] s = bf.readLine().split(" "); // x 좌표
    			for(int i=0;i<N;i++){
    				xy[i][0] = Long.parseLong(s[i]);
    			}
    			String[] s2 = bf.readLine().split(" "); // y 좌표
    			for(int i=0;i<N;i++){
    				xy[i][1] = Long.parseLong(s2[i]);
    			}
    			double E = Double.parseDouble(bf.readLine());// 환경 부담 세율
    			
    			// 각 정점별 인접리스트
    			ArrayList<Node>[] adjList = new ArrayList[N]; 
    			for(int i=0;i<N;i++) { // 좌표 값으로 가중치 만들자
    				adjList[i] = new ArrayList<>();
    				for(int j=0;j<N;j++) {
    					if(i==j) continue;
    					long L = (long)Math.pow(xy[i][0]-xy[j][0],2) + (long)Math.pow(xy[i][1]-xy[j][1],2); // 제곱이 문제 조건
    					adjList[i].add(new Node(j, L));
    				}
    			}
    			// 프림 알고리즘에 필요한 구조
    			long result = 0; // 최소 신장 트리 비용 누적
    			
    			PriorityQueue<Node> pQueue = new PriorityQueue<>();
    			pQueue.offer(new Node(0, 0)); // 0번 노드부터 시작하도록 !
    			
    			int cnt = 0; // 신장트리에 추가된 정점 수
    			while(true) { // 큐가 비어있지않을때랑 정점이 다 연결되면 break
    				
    			// step1. 신장트리의 구성에 포함되지 않은 정점 중 최소 비용 정점 선택
    			Node minVertex = pQueue.poll();
    			
    			if(visited[minVertex.vertex]) continue;
    
    			// step2. 신장트리에 추가
    			visited[minVertex.vertex] = true;
    			result += minVertex.weight;
    			if(++cnt==N) break;
    			
    			// step3. 신장트리에 새롭게 추가되는 정점과 신장트리에 포함되지 않은 정점들의 기존 최소 비용과 비교해서 갱신
    			//		    신장트트리에 새롭게 추가되는 정점의 모든 인접 정점을 들여다보며 처리
    			for(Node node: adjList[minVertex.vertex]) {
    				if(!visited[node.vertex]) {
    					pQueue.add(node);	
    				}
    			}
    		}
    		// 소수점 첫째자리에서 반올림
    		System.out.println("#"+tc+" "+Math.round(result*E));
    			
    		}// test case end
    	} // main end
    
    }

    adjMatrix.java

    public class SWEA1251_하나로_프림인접행렬 {
    
    	// 정점 중심
    	static class Node implements Comparable<Node>{
    		int vertex; // 노드
    		long weight; // 가중치
    		public Node(int vertex, long weight) {
    			super();
    			this.vertex = vertex;
    			this.weight = weight;
    		}
    		@Override
    		public int compareTo(Node o) {
    			return Long.compare(this.weight, o.weight);
    		}
    	}
    
    	public static void main(String[] args) throws IOException { // 0825 오전 라이브 강의 인접리스트로 prim 구현 참고 
    		BufferedReader bf = new BufferedReader(new InputStreamReader(System.in));
    		int T = Integer.parseInt(bf.readLine()); // 테스트 케이스 개수
    		for(int tc=1;tc<=T;tc++) {
    			int N = Integer.parseInt(bf.readLine()); // 정점의 수
    			long[][] adjMatrix = new long[N][N];
    			boolean[] visited = new boolean[N]; // 신장트리에 포함 여부
    			
    			long[][] xy = new long[N][2]; // x y 입력받기
    			String[] s = bf.readLine().split(" "); // x 좌표
    			for(int i=0;i<N;i++){
    				xy[i][0] = Long.parseLong(s[i]);
    			}
    			String[] s2 = bf.readLine().split(" "); // y 좌표
    			for(int i=0;i<N;i++){
    				xy[i][1] = Long.parseLong(s2[i]);
    			}
    			double E = Double.parseDouble(bf.readLine());// 환경 부담 세율
    			
    			// 인접 행렬로 구현 !
    			for(int i=0;i<N;i++) { // 각 섬끼리의 거리 저장
    				for(int j=0;j<N;j++) {
    					adjMatrix[i][j] = (long)Math.pow(xy[i][0]-xy[j][0],2) + (long)Math.pow(xy[i][1]-xy[j][1],2); // 제곱이 문제 조건
    					adjMatrix[j][i] = adjMatrix[i][j];
    				}
    			}
    			// 프림 알고리즘에 필요한 구조
    			long[] minEdge = new long[N];
    			// 프림 알고리즘  초기 상태 : 모든 점 최대값으로 설정.
    			Arrays.fill(minEdge, Long.MAX_VALUE);
    			minEdge[0]= 0; //임의의 점인 0번 인덱스를 시작점으로 만들어주기
    			
    			PriorityQueue<Node> q = new PriorityQueue<>();
                q.offer(new Node(0, minEdge[0]));
    
    			
    			long result = 0; // 최소 신장 트리 비용 누적
    			int cnt = 0; // 신장트리에 추가된 정점 수
    			while(true) {
    				
    				Node node = q.poll();
    				
    				if (visited[node.vertex]) {
                        continue;
                    }
                    visited[node.vertex] = true;
                    result += node.weight;
                    if (++cnt == N) {
                        break;
                    }
    
                    for (int i = 0; i < N; i++) {
                        if (!visited[i] && adjMatrix[node.vertex][i] < minEdge[i]) {
                        	minEdge[i] = adjMatrix[node.vertex][i];
                            q.offer(new Node(i, minEdge[i]));
                        }
                    }
                }
    				
    		// 소수점 첫째자리에서 반올림
    		System.out.println("#"+tc+" "+Math.round(result*E));
    			
    		}// test case end
    	} // main end
    
    }

     

    반응형

    댓글

Designed by SooJI