알고리즘/C++

[백준] 2064 - IP주소

엑츄얼리 2022. 12. 16. 12:51

https://www.acmicpc.net/problem/2064

 

2064번: IP 주소

네트워크에 연결되어 있는 컴퓨터들은 각각 하나의 IP 주소를 갖게 된다. 그리고 이러한 IP 주소를 갖는 컴퓨터들이 여러 개 모여서 하나의 IP 네트워크를 구성하게 된다. IP 네트워크는 ‘네트워

www.acmicpc.net

비트마스킹 방식을 사용하지 않으면 구조적으로 시간 초과가 날 수밖에 없는 문제였다.

 

알고리즘 방식

1. C++에는 Split함수가 없으므로 직접 구현

2. 입력된 IP주소를 '.'을 기준으로 Split 하여 4개의 숫자를 vector<vector<int>> ip에 저장

3. 네트워크 주소는 32-m자리까지 고정이고, 이후가 랜덤 값이므로,

    입력된 ip주소들 간에 차이가 발생하는 최소 자리수를 구한다.

4. 3.에서 구한 자릿수 직전까지가 네트워크 주소이고, 이를 기반으로 네트워크 마스크까지 구할 수 있다.

 

소스코드

/*
Date : 22.12.11
Time : 약 2시간...?
Comment :
	시간줄이자고 4중for문 사용했는데, 그냥 하는게 나았으려나 싶기도하고
	익숙하지 않다보니까 머리가 잘 안굴러간다 휴....

*/

#include <iostream>
#include <string> //stoi header
#include <vector>
#include <sstream> //stringstream, getline header

using namespace std;

int n;
string str;
vector<vector<int>> ip;

vector<int> split(string input, char delimeter) {
	vector<int> answer;
	stringstream ss(input);
	string temp;

	while (getline(ss, temp, delimeter)) {
		answer.push_back(stoi(temp));
	}
	return answer;
}

int threeBundle() {
	//.단위로 다른값이 있는지 확인
	for (int i = 0; i < 4; i++) {
		for (int j = 1; j < n; j++) {
			int result = ip[0][i] ^ ip[j][i];
			if (result != 0) {
				for (int k = 7; k >= 0; k--) {
					int mask = ip[0][i] & (1 << k);
					for (int l = 1; l < n; l++) {
						int buf = ip[l][i] & (1 << k);
						if (mask != buf) {
							return (8 * i) + (8 - k);
						}
					}
				}
			}
		}
	}
	return 0;
}

void answer(int position) {
	string networkAddress = "";
	string networkMask = "";
	if (position == 0) {
		int mask = (1 << 8) - 1;
		for (int i = 0; i < 4; i++) {
			networkAddress += to_string(ip[0][i]);
			networkMask += to_string(mask);
			if (i != 3) {
				networkAddress += '.';
				networkMask += '.';
			}
		}
	}
	else {
		//int mask = ~((1 << (32 + 1 - position)) - 1);
		int mask = ~((1 << (8 - (position - 1) % 8)) - 1);
		//cout << "mask : " << mask << '\n';
		for (int i = 0; i < 4; i++) {
			if (position > 8 * (i + 1)) {
				networkAddress += to_string(ip[0][i]);
				networkMask += "255";
			}
			else if (position <= 8 * i) {
				networkAddress += '0';
				networkMask += '0';
			}
			else {
				networkAddress += to_string(ip[0][i] & mask);
				networkMask += to_string((255 & mask));
			}

			if (i != 3) {
				networkAddress += '.';
				networkMask += '.';
			}
		}
	}

	cout << networkAddress << '\n' << networkMask;
}

int main(void) {
	ios::sync_with_stdio(false);
	cin.tie(NULL);

	cin >> n;

	//ip주소 모두 저장
	for (int i = 0; i < n; i++) {
		cin >> str;
		ip.push_back(split(str, '.'));
	}

	int position = threeBundle();
	//cout << "position : " << position << '\n';
	answer(position);
}