백준 5076(Web Pages)

jh Seo·2022년 11월 3일
0

백준

목록 보기
65/194
post-custom-banner

개요

백준 5076번: Web Pages

  • 입력
    Input will consist of a number of lines of HTML code, each line containing from 0 to 255 characters. The last line will contain a single # character – do not process this line. Within the text of each line will be zero or more tags. No angle bracket will be present unless it is part of a properly formed tag.

    Determine whether or not the HTML meets the rules specified above.

  • 출력
    Output will consist of a single line for each line of input. The line will contain either the word legal, or the word illegal.

접근 방식

  1. 기본적인 방식은 스택을 이용해 <> 안의 단어를 스택에 집어넣은 후
    </> 안의 단어를 스택의 top과 비교해 다르면 illegal을 출력하는 방식이다.

  2. 몇가지 주의할점은
    -> 반복문이 끝났을 때, 스택에 단어가 남아있다면 illegal,
    -> 문자 뒤에 /이 오면 무시,
    -> c++이다보니 string으로 값받고 char형으로 나누고, string값에 다시 덧붙이고
    하는 과정이 복잡하긴 하다.
    -> 스택을 매 함수마다 초기화! 중요하다 이거땜에 예제도 틀렸었다,

코드

#include<iostream>
#include<stack>
#include<string>

using namespace std;

bool checkIfProperlyNested(string inputStr);
stack<string> s;

void input() {
	string inputStr;
	while (1) {
		////버퍼에 남아있는 엔터 제거
		//cin.ignore();
		getline(cin,inputStr);
		if (inputStr == "#") {
			return;
		}
		if (checkIfProperlyNested(inputStr)) cout << "legal" << '\n';
		else cout << "illegal" << '\n';

	}
}
/// <summary>
/// string형 매개변수인 inputStr의 태그가 적절한지 비교하는 함수
/// </summary>
/// <param name="inputStr"></param>
/// <returns>태그가 적절하다면 true , 적절하지 않다면 else</returns>
bool checkIfProperlyNested(string inputStr) {
	while (!s.empty()) {
		s.pop();
	}
	//문자열의 각 char형 원소마다 비교 실행
	for (int i = 0; i < inputStr.length(); i++) {
		string tmp = "";
		//원소가 <라면 태그 여는 값
		if (inputStr[i] == '<') {
			//태그 다음 값 조사
			i++;
			//<다음 값이 /라면
			if (inputStr[i] == '/') {
				i++;
				//> 값이 나올때까지
				while (inputStr[i] != '>') {
					//임시 string변수 tmp에 더해준다.
					tmp += inputStr[i++];
				}
				//반복문 끝났다면 tmp값에 < >사이에 해당하는 단어들어있다.
				//스택이 비어있다면 짝이 안맞으므로 false 리턴
				if (s.empty()) {
					return false;
				}
				//스택이 차있다면
				else {
					//스택의 top값이 tmp값과 같다면
					if (s.top() == tmp) {
						s.pop();
					}
					//일치하지 않다면 false리턴
					else {
						return false;
					}
				}
			}
			//< 다음에 /가 아니라 처음 여는 꺽쇠라면
			else {
				while (inputStr[i] != ' ' && inputStr[i] != '>') {
					tmp += inputStr[i++];
				}
				//빈칸이 있을 경우 
				if (inputStr[i] == ' ') {
					//다음칸이 /라면 패스
					if (inputStr[++i] == '/') {
						//>를 가리키도록 
						i++;
						continue;
					}
					//다음칸에 주소가 나온다면
					//닫는 꺽쇠나올때까지 i증가
					while (inputStr[i] != '>') {
						i++;
					}
				}
				//처음 여는 꺽쇠이므로 스택에 푸시
				s.push(tmp);
			}
		}
	}
	if (!s.empty()) return false;
	return true;
}
int main() {
	input();
}

생각

자료구조 초기화를 또 생각을 못 했다!!

profile
코딩 창고!
post-custom-banner

0개의 댓글