알고리즘/C++
[백준] 2064 - IP주소
엑츄얼리
2022. 12. 16. 12:51
https://www.acmicpc.net/problem/2064
비트마스킹 방식을 사용하지 않으면 구조적으로 시간 초과가 날 수밖에 없는 문제였다.
알고리즘 방식
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);
}