아름다운 수프-퀵 가이드

오늘날 우리는 무료로 사용할 수있는 수많은 비정형 데이터 / 정보 (대부분 웹 데이터)를 보유하고 있습니다. 무료로 사용할 수있는 데이터는 읽기 쉽고 읽기 쉽지 않을 때도 있습니다. 데이터를 사용할 수있는 방법에 관계없이 웹 스크래핑은 구조화되지 않은 데이터를 읽기 및 분석하기 쉬운 구조화 된 데이터로 변환하는 데 매우 유용한 도구입니다. 즉,이 방대한 양의 데이터를 수집, 구성 및 분석하는 한 가지 방법은 웹 스크래핑입니다. 먼저 웹 스크래핑이 무엇인지 이해합시다.

웹 스크래핑이란 무엇입니까?

스크래핑은 단순히 데이터를 추출 (다양한 수단에서), 복사 및 선별하는 프로세스입니다.

웹 (웹 페이지 또는 웹 사이트 등)에서 데이터 또는 피드를 스크랩하거나 추출 할 때이를 웹 스크래핑이라고합니다.

따라서 웹 데이터 추출 또는 웹 수확이라고도하는 웹 스크래핑은 웹에서 데이터를 추출하는 것입니다. 간단히 말해서 웹 스크래핑은 개발자가 인터넷에서 데이터를 수집하고 분석 할 수있는 방법을 제공합니다.

웹 스크래핑이 필요한 이유

웹 스크래핑은 사람이 탐색하는 동안 수행하는 대부분의 작업을 자동화하는 훌륭한 도구 중 하나를 제공합니다. 웹 스크래핑은 기업에서 다양한 방식으로 사용됩니다.

연구용 데이터

스마트 분석가 (예 : 연구원 또는 저널리스트)는 웹 사이트에서 데이터를 수동으로 수집하고 정리하는 대신 웹 스크레이퍼를 사용합니다.

제품 가격 및 인기 비교

현재 웹 스크레이퍼를 사용하여 수많은 온라인 사이트에서 데이터를 수집하고 제품 인기도와 가격을 비교하는 데 사용하는 몇 가지 서비스가 있습니다.

SEO 모니터링

경쟁 분석 및 고객 웹 사이트에서 데이터를 가져 오는 데 사용되는 Ahrefs, Seobility, SEMrush 등과 같은 수많은 SEO 도구가 있습니다.

검색 엔진

웹 스크래핑에만 의존하는 비즈니스를하는 대형 IT 회사가 있습니다.

영업 및 마케팅

웹 스크래핑을 통해 수집 된 데이터는 마케팅 담당자가 다양한 틈새 시장과 경쟁 업체를 분석하는 데 사용하거나 콘텐츠 마케팅 또는 소셜 미디어 프로모션 서비스를 판매하는 영업 전문가가 사용할 수 있습니다.

왜 웹 스크랩 핑을위한 파이썬인가?

Python은 대부분의 웹 크롤링 관련 작업을 매우 쉽게 처리 할 수 ​​있으므로 웹 스크래핑에 가장 많이 사용되는 언어 중 하나입니다.

다음은 웹 스크래핑을 위해 Python을 선택하는 이유에 대한 몇 가지 요점입니다.

사용의 용이성

대부분의 개발자가 파이썬은 코드 작성이 매우 쉽다는 데 동의합니다. 중괄호 "{}"또는 세미콜론 ";"을 사용할 필요가 없습니다. 웹 스크레이퍼를 개발하는 동안 더 읽기 쉽고 사용하기 쉽습니다.

거대한 도서관 지원

Python은 다양한 요구 사항에 맞는 방대한 라이브러리 세트를 제공하므로 웹 스크래핑은 물론 데이터 시각화, 기계 학습 등에 적합합니다.

쉽게 설명 할 수있는 구문

Python 구문은 이해하기 쉽기 때문에 Python은 매우 읽기 쉬운 프로그래밍 언어입니다. Python은 매우 표현력이 뛰어나고 코드 들여 쓰기는 사용자가 코드에서 다른 블록이나 스쿠프를 구별하는 데 도움이됩니다.

동적으로 유형화 된 언어

Python은 동적 유형의 언어입니다. 즉, 변수에 할당 된 데이터가 변수 유형을 알려줍니다. 많은 시간을 절약하고 작업 속도를 높일 수 있습니다.

거대한 커뮤니티

Python 커뮤니티는 거대하므로 코드를 작성하는 동안 어디에 있든 도움이됩니다.

아름다운 수프 소개

Beautiful Soup은 "이상한 나라의 앨리스"에서 같은 이름의 루이스 캐롤시의 이름을 따서 명명 된 파이썬 라이브러리입니다. Beautiful Soup은 python 패키지이며 이름에서 알 수 있듯이 원하지 않는 데이터를 구문 분석하고 잘못된 HTML을 수정하여 지저분한 웹 데이터를 구성하고 형식을 지정하는 데 도움을주고 쉽게 이동할 수있는 XML 구조로 제공합니다.

간단히 말해, Beautiful Soup은 HTML 및 XML 문서에서 데이터를 가져올 수있는 파이썬 패키지입니다.

BeautifulSoup은 표준 파이썬 라이브러리가 아니므로 먼저 설치해야합니다. 우리는 최신 라이브러리 인 BeautifulSoup 4 라이브러리 (BS4라고도 함)를 설치할 것입니다.

기존 설정을 방해하지 않도록 작업 환경을 격리하기 위해 먼저 가상 환경을 만듭니다.

가상 환경 만들기 (선택 사항)

가상 환경을 사용하면 외부 설정에 영향을주지 않고 특정 프로젝트에 대해 격리 된 Python 작업 복사본을 만들 수 있습니다.

파이썬 패키지 머신을 설치하는 가장 좋은 방법은 pip를 사용하는 것입니다. 그러나 pip가 아직 설치되어 있지 않다면 (명령이나 셸 프롬프트에서 –“pip –version”을 사용하여 확인할 수 있습니다), 아래 명령을 제공하여 설치할 수 있습니다.

Linux 환경

$sudo apt-get install python-pip

Windows 환경

Windows에 pip를 설치하려면 다음을 수행하십시오.

  • get-pip.py를 https://bootstrap.pypa.io/get-pip.py 또는 github에서 컴퓨터로.

  • 명령 프롬프트를 열고 get-pip.py 파일이 포함 된 폴더로 이동합니다.

  • 다음 명령을 실행하십시오-

>python get-pip.py

이제 pip가 Windows 컴퓨터에 설치되었습니다.

아래 명령을 실행하여 설치된 pip를 확인할 수 있습니다.

>pip --version
pip 19.2.3 from c:\users\yadur\appdata\local\programs\python\python37\lib\site-packages\pip (python 3.7)

가상 환경 설치

명령 프롬프트에서 아래 명령을 실행하십시오-

>pip install virtualenv

실행 후 아래 스크린 샷을 볼 수 있습니다.

아래 명령은 현재 디렉토리에 가상 환경 ( "myEnv")을 생성합니다.

>virtualenv myEnv

스크린 샷

가상 환경을 활성화하려면 다음 명령을 실행하십시오.

>myEnv\Scripts\activate

위의 스크린 샷에서 접두사로 "myEnv"를 사용하여 가상 환경 "myEnv"에 있음을 알 수 있습니다.

가상 환경에서 나오려면 비활성화를 실행하십시오.

(myEnv) C:\Users\yadur>deactivate
C:\Users\yadur>

가상 환경이 준비되었으므로 이제 beautifulsoup을 설치하겠습니다.

BeautifulSoup 설치

BeautifulSoup은 표준 라이브러리가 아니기 때문에 설치해야합니다. BeautifulSoup 4 패키지 (BS4로 알려짐)를 사용할 것입니다.

Linux 머신

시스템 패키지 관리자를 사용하여 Debian 또는 Ubuntu linux에 bs4를 설치하려면 아래 명령을 실행하십시오.

$sudo apt-get install python-bs4 (for python 2.x)
$sudo apt-get install python3-bs4 (for python 3.x)

easy_install 또는 pip를 사용하여 bs4를 설치할 수 있습니다 (시스템 패키저를 사용하여 설치하는 데 문제가있는 경우).

$easy_install beautifulsoup4
$pip install beautifulsoup4

(python3을 사용하는 경우 각각 easy_install3 또는 pip3를 사용해야 할 수 있습니다.)

Windows 머신

Windows에 beautifulsoup4를 설치하는 것은 매우 간단합니다. 특히 pip가 이미 설치되어있는 경우 더욱 그렇습니다.

>pip install beautifulsoup4

이제 beautifulsoup4가 우리 컴퓨터에 설치되었습니다. 설치 후 발생한 몇 가지 문제에 대해 이야기하겠습니다.

설치 후 문제

Windows 시스템에서 발생할 수있는 잘못된 버전은 주로 다음을 통해 오류를 설치합니다.

  • 오류: ImportError “No module named HTMLParser”, 그러면 Python 3에서 python 2 버전의 코드를 실행해야합니다.

  • 오류: ImportError “No module named html.parser” 오류가 발생하면 Python 2에서 Python 3 버전의 코드를 실행해야합니다.

위의 두 가지 상황에서 벗어나는 가장 좋은 방법은 BeautifulSoup을 다시 설치하여 기존 설치를 완전히 제거하는 것입니다.

당신이 얻는 경우 SyntaxError “Invalid syntax” ROOT_TAG_NAME = u '[document]'라인에서, 패키지를 설치하여 파이썬 2 코드를 파이썬 3으로 변환해야합니다.

$ python3 setup.py install

또는 bs4 디렉토리에서 파이썬의 2 ~ 3 변환 스크립트를 수동으로 실행하여-

$ 2to3-3.2 -w bs4

파서 설치

기본적으로 Beautiful Soup은 Python의 표준 라이브러리에 포함 된 HTML 파서를 지원하지만 lxml 파서 또는 html5lib 파서와 같은 많은 외부 타사 Python 파서를 지원합니다.

lxml 또는 html5lib 파서를 설치하려면 다음 명령을 사용하십시오.

Linux 머신

$apt-get install python-lxml
$apt-get insall python-html5lib

Windows 머신

$pip install lxml
$pip install html5lib

일반적으로 사용자는 속도를 위해 lxml을 사용하며, python 2 (2.7.3 이전 버전) 또는 python 3 (3.2.2 이전)의 이전 버전을 사용하는 경우에는 lxml 또는 html5lib 파서를 사용하는 것이 좋습니다. 이전 버전을 처리하는 데별로 좋지 않습니다.

아름다운 수프 달리기

이제 html 페이지 중 하나에서 Beautiful Soup 패키지를 테스트 할 시간입니다 (웹 페이지 가져 오기 – https://www.tutorialspoint.com/index.htm, 원하는 다른 웹 페이지를 선택할 수 있습니다.)에서 일부 정보를 추출 할 수 있습니다.

아래 코드에서 웹 페이지에서 제목을 추출하려고합니다.

from bs4 import BeautifulSoup
import requests
url = "https://www.tutorialspoint.com/index.htm"
req = requests.get(url)
soup = BeautifulSoup(req.text, "html.parser")
print(soup.title)

산출

<title>H2O, Colab, Theano, Flutter, KNime, Mean.js, Weka, Solidity, Org.Json, AWS QuickSight, JSON.Simple, Jackson Annotations, Passay, Boon, MuleSoft, Nagios, Matplotlib, Java NIO, PyTorch, SLF4J, Parallax Scrolling, Java Cryptography</title>

한 가지 일반적인 작업은 웹 페이지 내의 모든 URL을 추출하는 것입니다. 이를 위해 아래 코드 줄을 추가하면됩니다.

for link in soup.find_all('a'):
print(link.get('href'))

산출

https://www.tutorialspoint.com/index.htm
https://www.tutorialspoint.com/about/about_careers.htm
https://www.tutorialspoint.com/questions/index.php
https://www.tutorialspoint.com/online_dev_tools.htm
https://www.tutorialspoint.com/codingground.htm
https://www.tutorialspoint.com/current_affairs.htm
https://www.tutorialspoint.com/upsc_ias_exams.htm
https://www.tutorialspoint.com/tutor_connect/index.php
https://www.tutorialspoint.com/whiteboard.htm
https://www.tutorialspoint.com/netmeeting.php
https://www.tutorialspoint.com/index.htm
https://www.tutorialspoint.com/tutorialslibrary.htm
https://www.tutorialspoint.com/videotutorials/index.php
https://store.tutorialspoint.com
https://www.tutorialspoint.com/gate_exams_tutorials.htm
https://www.tutorialspoint.com/html_online_training/index.asp
https://www.tutorialspoint.com/css_online_training/index.asp
https://www.tutorialspoint.com/3d_animation_online_training/index.asp
https://www.tutorialspoint.com/swift_4_online_training/index.asp
https://www.tutorialspoint.com/blockchain_online_training/index.asp
https://www.tutorialspoint.com/reactjs_online_training/index.asp
https://www.tutorix.com
https://www.tutorialspoint.com/videotutorials/top-courses.php
https://www.tutorialspoint.com/the_full_stack_web_development/index.asp
….
….
https://www.tutorialspoint.com/online_dev_tools.htm
https://www.tutorialspoint.com/free_web_graphics.htm
https://www.tutorialspoint.com/online_file_conversion.htm
https://www.tutorialspoint.com/netmeeting.php
https://www.tutorialspoint.com/free_online_whiteboard.htm
http://www.tutorialspoint.com
https://www.facebook.com/tutorialspointindia
https://plus.google.com/u/0/+tutorialspoint
http://www.twitter.com/tutorialspoint
http://www.linkedin.com/company/tutorialspoint
https://www.youtube.com/channel/UCVLbzhxVTiTLiVKeGV7WEBg
https://www.tutorialspoint.com/index.htm
/about/about_privacy.htm#cookies
/about/faq.htm
/about/about_helping.htm
/about/contact_us.htm

마찬가지로 beautifulsoup4를 사용하여 유용한 정보를 추출 할 수 있습니다.

이제 위의 예에서 "수프"에 대해 더 많이 이해하겠습니다.

이전 코드 예제에서는 문자열 메서드를 사용하여 아름다운 생성자를 통해 문서를 구문 분석했습니다. 또 다른 방법은 열린 파일 핸들을 통해 문서를 전달하는 것입니다.

from bs4 import BeautifulSoup
with open("example.html") as fp:
   soup = BeautifulSoup(fp)
soup = BeautifulSoup("<html>data</html>")

먼저 문서가 유니 코드로 변환되고 HTML 엔티티가 유니 코드 문자로 변환됩니다. </ p>

import bs4
html = '''<b>tutorialspoint</b>, <i>&web scraping &data science;</i>'''
soup = bs4.BeautifulSoup(html, 'lxml')
print(soup)

산출

<html><body><b>tutorialspoint</b>, <i>&web scraping &data science;</i></body></html>

그런 다음 BeautifulSoup은 HTML 파서를 사용하여 데이터를 파싱하거나 XML 파서를 사용하여 파싱하도록 명시 적으로 지시합니다.

HTML 트리 구조

HTML 페이지의 다른 구성 요소를 살펴보기 전에 먼저 HTML 트리 구조를 이해하겠습니다.

문서 트리의 루트 요소는 부모, 자식 및 형제를 가질 수있는 html이며 트리 구조에서의 위치에 따라 결정됩니다. HTML 요소, 속성 및 텍스트 사이를 이동하려면 트리 구조에서 노드 사이를 이동해야합니다.

웹 페이지가 아래와 같다고 가정 해 보겠습니다.

다음과 같이 html 문서로 번역됩니다-

<html><head><title>TutorialsPoint</title></head><h1>Tutorialspoint Online Library</h1><p<<b>It's all Free</b></p></body></html>

즉, 위의 html 문서의 경우 다음과 같은 html 트리 구조가 있습니다.

html 문서 또는 문자열을 beautifulsoup 생성자에 전달하면 beautifulsoup은 기본적으로 복잡한 html 페이지를 다른 파이썬 객체로 변환합니다. 아래에서는 네 가지 주요 유형의 객체에 대해 설명합니다.

  • Tag

  • NavigableString

  • BeautifulSoup

  • Comments

태그 개체

HTML 태그는 다양한 유형의 콘텐츠를 정의하는 데 사용됩니다. BeautifulSoup의 태그 객체는 실제 페이지 나 문서의 HTML 또는 XML 태그에 해당합니다.

>>> from bs4 import BeautifulSoup
>>> soup = BeautifulSoup('<b class="boldest">TutorialsPoint</b>')
>>> tag = soup.html
>>> type(tag)
<class 'bs4.element.Tag'>

태그에는 많은 속성과 메서드가 포함되어 있으며 태그의 두 가지 중요한 기능은 이름과 속성입니다.

이름 (tag.name)

모든 태그에는 이름이 포함되어 있으며 접미사로 '.name'을 통해 액세스 할 수 있습니다. tag.name은 태그 유형을 반환합니다.

>>> tag.name
'html'

그러나 태그 이름을 변경하면 BeautifulSoup에서 생성 한 HTML 마크 업에도 동일하게 반영됩니다.

>>> tag.name = "Strong"
>>> tag
<Strong><body><b class="boldest">TutorialsPoint</b></body></Strong>
>>> tag.name
'Strong'

속성 (tag.attrs)

태그 개체는 여러 속성을 가질 수 있습니다. 태그 <b class =”boldest”>에는 값이“boldest”인 'class'속성이 있습니다. 태그가 아닌 모든 것은 기본적으로 속성이며 값을 포함해야합니다. 키에 액세스 (위의 예에서 "클래스"에 액세스)하거나 ".attrs"를 통해 직접 액세스하여 속성에 액세스 할 수 있습니다.

>>> tutorialsP = BeautifulSoup("<div class='tutorialsP'></div>",'lxml')
>>> tag2 = tutorialsP.div
>>> tag2['class']
['tutorialsP']

태그의 속성 (추가 / 제거 / 수정)에 대한 모든 종류의 수정을 수행 할 수 있습니다.

>>> tag2['class'] = 'Online-Learning'
>>> tag2['style'] = '2007'
>>>
>>> tag2
<div class="Online-Learning" style="2007"></div>
>>> del tag2['style']
>>> tag2
<div class="Online-Learning"></div>
>>> del tag['class']
>>> tag
<b SecondAttribute="2">TutorialsPoint</b>
>>>
>>> del tag['SecondAttribute']
>>> tag
</b>
>>> tag2['class']
'Online-Learning'
>>> tag2['style']
KeyError: 'style'

다중 값 속성

HTML5 속성 중 일부는 여러 값을 가질 수 있습니다. 가장 일반적으로 사용되는 것은 여러 CSS 값을 가질 수있는 클래스 속성입니다. 기타에는 'rel', 'rev', 'headers', 'accesskey'및 'accept-charset'이 있습니다. 아름다운 수프의 다중 값 속성이 목록으로 표시됩니다.

>>> from bs4 import BeautifulSoup
>>>
>>> css_soup = BeautifulSoup('<p class="body"></p>')
>>> css_soup.p['class']
['body']
>>>
>>> css_soup = BeautifulSoup('<p class="body bold"></p>')
>>> css_soup.p['class']
['body', 'bold']

그러나 어떤 속성에 둘 이상의 값이 포함되어 있지만 HTML 표준의 모든 버전에서 다중 값 속성이 아닌 경우 아름다운 수프는 속성을 그대로 둡니다.

>>> id_soup = BeautifulSoup('<p id="body bold"></p>')
>>> id_soup.p['id']
'body bold'
>>> type(id_soup.p['id'])
<class 'str'>

태그를 문자열로 바꾸면 여러 속성 값을 통합 할 수 있습니다.

>>> rel_soup = BeautifulSoup("<p> tutorialspoint Main <a rel='Index'> Page</a></p>")
>>> rel_soup.a['rel']
['Index']
>>> rel_soup.a['rel'] = ['Index', ' Online Library, Its all Free']
>>> print(rel_soup.p)
<p> tutorialspoint Main <a rel="Index Online Library, Its all Free"> Page</a></p>

'get_attribute_list'를 사용하면 다중 값인지 여부에 관계없이 항상 목록, 문자열 인 값을 얻을 수 있습니다.

id_soup.p.get_attribute_list(‘id’)

그러나 문서를 'xml'로 구문 분석하면 다중 값 속성이 없습니다.

>>> xml_soup = BeautifulSoup('<p class="body bold"></p>', 'xml')
>>> xml_soup.p['class']
'body bold'

NavigableString

navigablestring 객체는 태그의 내용을 나타내는 데 사용됩니다. 내용에 액세스하려면 ".string"을 태그와 함께 사용하십시오.

>>> from bs4 import BeautifulSoup
>>> soup = BeautifulSoup("<h2 id='message'>Hello, Tutorialspoint!</h2>")
>>>
>>> soup.string
'Hello, Tutorialspoint!'
>>> type(soup.string)
>

문자열을 다른 문자열로 바꿀 수 있지만 기존 문자열을 편집 할 수는 없습니다.

>>> soup = BeautifulSoup("<h2 id='message'>Hello, Tutorialspoint!</h2>")
>>> soup.string.replace_with("Online Learning!")
'Hello, Tutorialspoint!'
>>> soup.string
'Online Learning!'
>>> soup
<html><body><h2 id="message">Online Learning!</h2></body></html>

BeautifulSoup

BeautifulSoup은 웹 리소스를 스크랩하려고 할 때 생성되는 개체입니다. 그래서 그것은 우리가 긁어 내려는 완전한 문서입니다. 대부분의 경우 태그 객체로 취급됩니다.

>>> from bs4 import BeautifulSoup
>>> soup = BeautifulSoup("<h2 id='message'>Hello, Tutorialspoint!</h2>")
>>> type(soup)
<class 'bs4.BeautifulSoup'>
>>> soup.name
'[document]'

코멘트

주석 개체는 웹 문서의 주석 부분을 보여줍니다. 특별한 유형의 NavigableString입니다.

>>> soup = BeautifulSoup('<p><!-- Everything inside it is COMMENTS --></p>')
>>> comment = soup.p.string
>>> type(comment)
<class 'bs4.element.Comment'>
>>> type(comment)
<class 'bs4.element.Comment'>
>>> print(soup.p.prettify())
<p>
<!-- Everything inside it is COMMENTS -->
</p>

NavigableString 객체

navigablestring 객체는 태그 자체가 아닌 태그 내의 텍스트를 나타내는 데 사용됩니다.

이 장에서는 태그로 탐색에 대해 설명합니다.

아래는 HTML 문서입니다.

>>> html_doc = """
<html><head><title>Tutorials Point</title></head>
<body>
<p class="title"><b>The Biggest Online Tutorials Library, It's all Free</b></p>
<p class="prog">Top 5 most used Programming Languages are:
<a href="https://www.tutorialspoint.com/java/java_overview.htm" class="prog" id="link1">Java</a>,
<a href="https://www.tutorialspoint.com/cprogramming/index.htm" class="prog" id="link2">C</a>,
<a href="https://www.tutorialspoint.com/python/index.htm" class="prog" id="link3">Python</a>,
<a href="https://www.tutorialspoint.com/javascript/javascript_overview.htm" class="prog" id="link4">JavaScript</a> and
<a href="https://www.tutorialspoint.com/ruby/index.htm" class="prog" id="link5">C</a>;
as per online survey.</p>
<p class="prog">Programming Languages</p>
"""
>>>
>>> from bs4 import BeautifulSoup
>>> soup = BeautifulSoup(html_doc, 'html.parser')
>>>

위의 문서를 바탕으로 문서의 한 부분에서 다른 부분으로 이동하려고합니다.

아래로

HTML 문서에서 중요한 요소 중 하나는 다른 태그 / 문자열 (태그의 자식)을 포함 할 수있는 태그입니다. Beautiful Soup은 태그의 자식을 탐색하고 반복하는 다양한 방법을 제공합니다.

태그 이름을 사용하여 탐색

구문 분석 트리를 검색하는 가장 쉬운 방법은 이름으로 태그를 검색하는 것입니다. <head> 태그를 원한다면 soup.head를 사용하십시오.

>>> soup.head
<head>&t;title>Tutorials Point</title></head>
>>> soup.title
<title>Tutorials Point</title>

<body> 태그에서 특정 태그 (예 : 첫 번째 <b> 태그)를 가져옵니다.

>>> soup.body.b
<b>The Biggest Online Tutorials Library, It's all Free</b>

태그 이름을 속성으로 사용하면 해당 이름의 첫 번째 태그 만 제공됩니다.

>>> soup.a
<a class="prog" href="https://www.tutorialspoint.com/java/java_overview.htm" id="link1">Java</a>

모든 태그의 속성을 얻으려면 find_all () 메소드를 사용할 수 있습니다.

>>> soup.find_all("a")
[<a class="prog" href="https://www.tutorialspoint.com/java/java_overview.htm" id="link1">Java</a>, <a class="prog" href="https://www.tutorialspoint.com/cprogramming/index.htm" id="link2">C</a>, <a class="prog" href="https://www.tutorialspoint.com/python/index.htm" id="link3">Python</a>, <a class="prog" href="https://www.tutorialspoint.com/javascript/javascript_overview.htm" id="link4">JavaScript</a>, <a class="prog" href="https://www.tutorialspoint.com/ruby/index.htm" id="link5">C</a>]>>> soup.find_all("a")
[<a class="prog" href="https://www.tutorialspoint.com/java/java_overview.htm" id="link1">Java</a>, <a class="prog" href="https://www.tutorialspoint.com/cprogramming/index.htm" id="link2">C</a>, <a class="prog" href="https://www.tutorialspoint.com/python/index.htm" id="link3">Python</a>, <a class="prog" href="https://www.tutorialspoint.com/javascript/javascript_overview.htm" id="link4">JavaScript</a>, <a class="prog" href="https://www.tutorialspoint.com/ruby/index.htm" id="link5">C</a>]

.contents 및 .children

.contents로 목록에서 태그의 자식을 검색 할 수 있습니다.

>>> head_tag = soup.head
>>> head_tag
<head><title>Tutorials Point</title></head>
>>> Htag = soup.head
>>> Htag
<head><title>Tutorials Point</title></head>
>>>
>>> Htag.contents
[<title>Tutorials Point</title>
>>>
>>> Ttag = head_tag.contents[0]
>>> Ttag
<title>Tutorials Point</title>
>>> Ttag.contents
['Tutorials Point']

BeautifulSoup 객체 자체에는 자식이 있습니다. 이 경우 <html> 태그는 BeautifulSoup 객체의 자식입니다.

>>> len(soup.contents)
2
>>> soup.contents[1].name
'html'

문자열은 아무것도 포함 할 수 없기 때문에 .contents가 없습니다.

>>> text = Ttag.contents[0]
>>> text.contents
self.__class__.__name__, attr))
AttributeError: 'NavigableString' object has no attribute 'contents'

목록으로 가져 오는 대신 .children 생성기를 사용하여 태그의 자식에 액세스합니다.

>>> for child in Ttag.children:
print(child)
Tutorials Point

.자손

.descendants 속성을 사용하면 태그의 모든 자식을 재귀 적으로 반복 할 수 있습니다.

직계 자식과 직계 자식 등

>>> for child in Htag.descendants:
print(child)
<title>Tutorials Point</title>
Tutorials Point

<head> 태그에는 자식이 하나만 있지만 하위 항목이 두 개 있습니다. <title> 태그와 <title> 태그의 자식입니다. beautifulsoup 객체는 오직 하나의 직계 자식 (<html> 태그) 만 가지고 있지만, 많은 자손이 있습니다.

>>> len(list(soup.children))
2
>>> len(list(soup.descendants))
33

.끈

태그에 자식이 하나만 있고 그 자식이 NavigableString이면 자식은 .string으로 사용할 수 있습니다.

>>> Ttag.string
'Tutorials Point'

태그의 유일한 자식이 다른 태그이고 해당 태그에 .string이있는 경우 부모 태그는 자식과 동일한 .string을 갖는 것으로 간주됩니다.

>>> Htag.contents
[<title>Tutorials Point</title>]
>>>
>>> Htag.string
'Tutorials Point'

그러나 태그에 하나 이상의 것이 포함되어 있으면 .string이 무엇을 참조해야하는지 명확하지 않으므로 .string은 None으로 정의됩니다.

>>> print(soup.html.string)
None

.strings 및 stripped_strings

태그 안에 하나 이상의 것이있는 경우에도 문자열 만 볼 수 있습니다. .strings 생성기를 사용하십시오-

>>> for string in soup.strings:
print(repr(string))
'\n'
'Tutorials Point'
'\n'
'\n'
"The Biggest Online Tutorials Library, It's all Free"
'\n'
'Top 5 most used Programming Languages are: \n'
'Java'
',\n'
'C'
',\n'
'Python'
',\n'
'JavaScript'
' and\n'
'C'
';\n \nas per online survey.'
'\n'
'Programming Languages'
'\n'

여분의 공백을 제거하려면 .stripped_strings 생성기를 사용하십시오.

>>> for string in soup.stripped_strings:
print(repr(string))
'Tutorials Point'
"The Biggest Online Tutorials Library, It's all Free"
'Top 5 most used Programming Languages are:'
'Java'
','
'C'
','
'Python'
','
'JavaScript'
'and'
'C'
';\n \nas per online survey.'
'Programming Languages'

올라간다

"가계도"비유에서 모든 태그와 모든 문자열에는 부모가 있습니다.

.부모의

요소의 상위 요소에 액세스하려면 .parent 속성을 사용하십시오.

>>> Ttag = soup.title
>>> Ttag
<title>Tutorials Point</title>
>>> Ttag.parent
<head>title>Tutorials Point</title></head>

html_doc에서 제목 문자열 자체에는 부모가 있습니다.

>>> Ttag.string.parent
<title>Tutorials Point</title>

<html>과 같은 최상위 태그의 부모는 Beautifulsoup 객체 자체입니다.

>>> htmltag = soup.html
>>> type(htmltag.parent)
<class 'bs4.BeautifulSoup'>

Beautifulsoup 객체의 부모는 None으로 정의됩니다-

>>> print(soup.parent)
None

.부모님

모든 상위 요소를 반복하려면 .parents 속성을 사용하십시오.

>>> link = soup.a
>>> link
<a class="prog" href="https://www.tutorialspoint.com/java/java_overview.htm" id="link1">Java</a>
>>>
>>> for parent in link.parents:
if parent is None:
print(parent)
else:
print(parent.name)
p
body
html
[document]

옆으로

아래는 간단한 문서입니다.

>>> sibling_soup = BeautifulSoup("<a><b>TutorialsPoint</b><c><strong>The Biggest Online Tutorials Library, It's all Free</strong></b></a>")
>>> print(sibling_soup.prettify())
<html>
<body>
   <a>
      <b>
         TutorialsPoint
      </b>
      <c>
         <strong>
            The Biggest Online Tutorials Library, It's all Free
         </strong>
      </c>
   </a>
</body>
</html>

위의 문서에서 <b> 및 <c> 태그는 동일한 수준에 있으며 둘 다 동일한 태그의 하위입니다. <b> 및 <c> 태그는 모두 형제입니다.

.next_sibling 및 .previous_sibling

.next_sibling 및 .previous_sibling을 사용하여 구문 분석 트리의 동일한 레벨에있는 페이지 요소 사이를 탐색하십시오.

>>> sibling_soup.b.next_sibling
<c><strong>The Biggest Online Tutorials Library, It's all Free</strong></c>
>>>
>>> sibling_soup.c.previous_sibling
<b>TutorialsPoint</b>

<b> 태그에는 .next_sibling이 있지만 .previous_sibling은 없습니다. 트리의 동일한 레벨에서 <b> 태그 앞에 아무것도 없기 때문에 동일한 대소 문자가 <c> 태그와 함께 사용됩니다.

>>> print(sibling_soup.b.previous_sibling)
None
>>> print(sibling_soup.c.next_sibling)
None

두 문자열은 동일한 부모가 없기 때문에 형제가 아닙니다.

>>> sibling_soup.b.string
'TutorialsPoint'
>>>
>>> print(sibling_soup.b.string.next_sibling)
None

.next_siblings 및 .previous_siblings

태그의 형제를 반복하려면 .next_siblings 및 .previous_siblings를 사용하십시오.

>>> for sibling in soup.a.next_siblings:
print(repr(sibling))
',\n'
<a class="prog" href="https://www.tutorialspoint.com/cprogramming/index.htm" id="link2">C</a>
',\n'
>a class="prog" href="https://www.tutorialspoint.com/python/index.htm" id="link3">Python</a>
',\n'
<a class="prog" href="https://www.tutorialspoint.com/javascript/javascript_overview.htm" id="link4">JavaScript</a>
' and\n'
<a class="prog" href="https://www.tutorialspoint.com/ruby/index.htm"
id="link5">C</a>
';\n \nas per online survey.'
>>> for sibling in soup.find(id="link3").previous_siblings:
print(repr(sibling))
',\n'
<a class="prog" href="https://www.tutorialspoint.com/cprogramming/index.htm" id="link2">C</a>
',\n'
<a class="prog" href="https://www.tutorialspoint.com/java/java_overview.htm" id="link1">Java</a>
'Top 5 most used Programming Languages are: \n'

앞뒤로

이제 이전 "html_doc"예제에서 처음 두 줄로 돌아가 보겠습니다.

&t;html><head><title>Tutorials Point</title></head>
<body>
<h4 class="tagLine"><b>The Biggest Online Tutorials Library, It's all Free</b></h4>

HTML 파서는 위의 문자열을 "open an <html> tag", "open an <head> tag", "open the <title> tag", "add a string"과 같은 일련의 이벤트로 변환합니다. "close the </ title> 태그", "close the </ head> 태그", "open a <h4> 태그"등. BeautifulSoup은 문서의 초기 구문 분석을 재구성하는 다양한 방법을 제공합니다.

.next_element 및 .previous_element

태그 또는 문자열의 .next_element 속성은 직후에 구문 분석 된 모든 것을 가리 킵니다. 때로는 .next_sibling과 비슷해 보이지만 완전히 동일하지는 않습니다. 다음은 "html_doc"예제 문서의 마지막 <a> 태그입니다.

>>> last_a_tag = soup.find("a", id="link5")
>>> last_a_tag
<a class="prog" href="https://www.tutorialspoint.com/ruby/index.htm" id="link5">C</a>
>>> last_a_tag.next_sibling
';\n \nas per online survey.'

그러나 <a> 태그 바로 뒤에 파싱 된 <a> 태그의 .next_element는 그 문장의 나머지 부분이 아닙니다. "C"라는 단어입니다.

>>> last_a_tag.next_element
'C'

위의 동작은 원래 마크 업에서 "C"문자가 세미콜론 앞에 나타나기 때문입니다. 파서가 <a> 태그, 문자 "C", 닫는 </a> 태그, 세미콜론 및 나머지 문장을 발견했습니다. 세미콜론은 <a> 태그와 동일한 수준에 있지만 문자 "C"가 먼저 발견되었습니다.

.previous_element 속성은 .next_element의 정반대입니다. 이 요소 바로 전에 구문 분석 된 요소를 가리 킵니다.

>>> last_a_tag.previous_element
' and\n'
>>>
>>> last_a_tag.previous_element.next_element
<a class="prog" href="https://www.tutorialspoint.com/ruby/index.htm" id="link5">C</a>

.next_elements 및 .previous_elements

이 반복기를 사용하여 요소로 앞뒤로 이동합니다.

>>> for element in last_a_tag.next_e lements:
print(repr(element))
'C'
';\n \nas per online survey.'
'\n'
<p class="prog">Programming Languages</p>
'Programming Languages'
'\n'

파스 트리를 검색 할 수있는 Beautifulsoup 메서드가 많이 있습니다. 가장 일반적이고 사용되는 두 가지 메서드는 find () 및 find_all ()입니다.

find () 및 find_all ()에 대해 이야기하기 전에 이러한 메서드에 전달할 수있는 여러 필터의 몇 가지 예를 살펴 보겠습니다.

필터 종류

이러한 메서드에 전달할 수있는 다른 필터가 있으며 이러한 필터를 검색 API 전체에서 반복해서 사용하므로 이러한 필터를 이해하는 것이 중요합니다. 태그의 이름, 속성, 문자열 텍스트 또는 이들을 혼합하여 이러한 필터를 사용할 수 있습니다.

문자열

가장 간단한 필터 유형 중 하나는 문자열입니다. 검색 방법과 Beautifulsoup에 문자열을 전달하면 해당 문자열에 대한 일치가 수행됩니다.

아래 코드는 문서의 모든 <p> 태그를 찾을 수 있습니다.

>>> markup = BeautifulSoup('<p>Top Three</p><p><pre>Programming Languages are:</pre></p><p><b>Java, Python, Cplusplus</b></p>')
>>> markup.find_all('p')
[<p>Top Three</p>, <p></p>, <p><b>Java, Python, Cplusplus</b></p>]

정규식

주어진 문자열 / 태그로 시작하는 모든 태그를 찾을 수 있습니다. 그 전에 정규 표현식을 사용하기 위해 re 모듈을 가져와야합니다.

>>> import re
>>> markup = BeautifulSoup('<p>Top Three</p><p><pre>Programming Languages are:</pre></p><p><b>Java, Python, Cplusplus</b></p>')
>>>
>>> markup.find_all(re.compile('^p'))
[<p>Top Three</p>, <p></p>, <pre>Programming Languages are:</pre>, <p><b>Java, Python, Cplusplus</b></p>]

명부

목록을 제공하여 찾을 여러 태그를 전달할 수 있습니다. 아래 코드는 모든 <b> 및 <pre> 태그를 찾습니다-

>>> markup.find_all(['pre', 'b'])
[<pre>Programming Languages are:</pre>, <b>Java, Python, Cplusplus</b>]

진실

True는 찾을 수있는 모든 태그를 반환하지만 자체적으로 문자열은 반환하지 않습니다.

>>> markup.find_all(True)
[<html><body><p>Top Three</p><p></p><pre>Programming Languages are:</pre>
<p><b>Java, Python, Cplusplus</b> </p> </body></html>, 
<body><p>Top Three</p><p></p><pre> Programming Languages are:</pre><p><b>Java, Python, Cplusplus</b></p>
</body>, 
<p>Top Three</p>, <p></p>, <pre>Programming Languages are:</pre>, <p><b>Java, Python, Cplusplus</b></p>, <b>Java, Python, Cplusplus</b>]

위의 수프에서 태그 만 반환하려면-

>>> for tag in markup.find_all(True):
(tag.name)
'html'
'body'
'p'
'p'
'pre'
'p'
'b'

find_all ()

find_all을 사용하여 페이지 응답에서 특정 태그의 모든 발생을 추출 할 수 있습니다.

통사론

find_all(name, attrs, recursive, string, limit, **kwargs)

IMDB-“최고 평점 영화”에서 흥미로운 데이터를 추출해 보겠습니다.

>>> url="https://www.imdb.com/chart/top/?ref_=nv_mv_250"
>>> content = requests.get(url)
>>> soup = BeautifulSoup(content.text, 'html.parser')
#Extract title Page
>>> print(soup.find('title'))
<title>IMDb Top 250 - IMDb</title>

#Extracting main heading
>>> for heading in soup.find_all('h1'):
   print(heading.text)
Top Rated Movies

#Extracting sub-heading
>>> for heading in soup.find_all('h3'):
   print(heading.text)
   
IMDb Charts
You Have Seen
   IMDb Charts
   Top India Charts
Top Rated Movies by Genre
Recently Viewed

위에서 우리는 find_all이 우리가 정의한 검색 기준과 일치하는 모든 항목을 제공하는 것을 볼 수 있습니다. find_all ()과 함께 사용할 수있는 모든 필터는 find () 및 find_parents () 또는 find_siblings ()와 같은 다른 검색 방법과 함께 사용할 수 있습니다.

찾기()

위에서 보았 듯이 find_all ()은 모든 내용을 찾기 위해 전체 문서를 스캔하는 데 사용됩니다. 요구 사항은 하나의 결과 만 찾는 것입니다. 문서에 <body> 태그가 하나만 포함되어 있다는 것을 알고 있다면 전체 문서를 검색하는 데 시간 낭비입니다. 한 가지 방법은 매번 limit = 1로 find_all ()을 호출하는 것입니다. 그렇지 않으면 find () 메서드를 사용하여 똑같이 할 수 있습니다.

통사론

find(name, attrs, recursive, string, **kwargs)

따라서 두 가지 다른 방법 아래에서 동일한 출력을 제공합니다.

>>> soup.find_all('title',limit=1)
[<title>IMDb Top 250 - IMDb</title>]
>>>
>>> soup.find('title')
<title>IMDb Top 250 - IMDb</title>

위의 출력에서 ​​find_all () 메서드는 단일 항목을 포함하는 목록을 반환하는 반면 find () 메서드는 단일 결과를 반환합니다.

find ()와 find_all () 메서드의 또 다른 차이점은-

>>> soup.find_all('h2')
[]
>>>
>>> soup.find('h2')

soup.find_all () 메서드가 아무것도 찾을 수 없으면 빈 목록을 반환하고 find ()는 None을 반환합니다.

find_parents () 및 find_parent ()

트리를 순회하는 find_all () 및 find () 메소드와 달리 태그의 하위 항목을 보면 find_parents () 및 find_parents methods ()는 그 반대를 수행하며 트리를 위쪽으로 순회하고 태그 (또는 문자열의) 부모를 봅니다.

통사론

find_parents(name, attrs, string, limit, **kwargs)
find_parent(name, attrs, string, **kwargs)

>>> a_string = soup.find(string="The Godfather")
>>> a_string
'The Godfather'
>>> a_string.find_parents('a')
[<a href="/title/tt0068646/" title="Francis Ford Coppola (dir.), Marlon Brando, Al Pacino">The Godfather</a>]
>>> a_string.find_parent('a')
<a href="/title/tt0068646/" title="Francis Ford Coppola (dir.), Marlon Brando, Al Pacino">The Godfather</a>
>>> a_string.find_parent('tr')
<tr>

<td class="posterColumn">
<span data-value="2" name="rk"></span>
<span data-value="9.149038526210072" name="ir"></span>
<span data-value="6.93792E10" name="us"></span>
<span data-value="1485540" name="nv"></span>
<span data-value="-1.850961473789928" name="ur"></span>
<a href="/title/tt0068646/"> <img alt="The Godfather" height="67" src="https://m.media-amazon.com/images/M/MV5BM2MyNjYxNmUtYTAwNi00MTYxLWJmNWYtYzZlODY3ZTk3OTFlXkEyXkFqcGdeQXVyNzkwMjQ5NzM@._V1_UY67_CR1,0,45,67_AL_.jpg" width="45"/>
</a> </td>
<td class="titleColumn">
2.
<a href="/title/tt0068646/" title="Francis Ford Coppola (dir.), Marlon Brando, Al Pacino">The Godfather</a>
<span class="secondaryInfo">(1972)</span>
</td>
<td class="ratingColumn imdbRating">
<strong title="9.1 based on 1,485,540 user ratings">9.1</strong>
</td>
<td class="ratingColumn">
<div class="seen-widget seen-widget-tt0068646 pending" data-titleid="tt0068646">
<div class="boundary">
<div class="popover">
<span class="delete"> </span><ol><li>1<li>2<li>3<li>4<li>5<li>6<li>7<li>8<li>9<li>10</li>0</li></li></li></li&td;</li></li></li></li></li></ol> </div>
</div>
<div class="inline">
<div class="pending"></div>
<div class="unseeable">NOT YET RELEASED</div>
<div class="unseen"> </div>
<div class="rating"></div>
<div class="seen">Seen</div>
</div>
</div>
</td>
<td class="watchlistColumn">

<div class="wlb_ribbon" data-recordmetrics="true" data-tconst="tt0068646"></div>
</td>
</tr>
>>>
>>> a_string.find_parents('td')
[<td class="titleColumn">
2.
<a href="/title/tt0068646/" title="Francis Ford Coppola (dir.), Marlon Brando, Al Pacino">The Godfather</a>
<span class="secondaryInfo">(1972)</span>
</td>]

8 개의 다른 유사한 방법이 있습니다-

find_next_siblings(name, attrs, string, limit, **kwargs)
find_next_sibling(name, attrs, string, **kwargs)

find_previous_siblings(name, attrs, string, limit, **kwargs)
find_previous_sibling(name, attrs, string, **kwargs)

find_all_next(name, attrs, string, limit, **kwargs)
find_next(name, attrs, string, **kwargs)

find_all_previous(name, attrs, string, limit, **kwargs)
find_previous(name, attrs, string, **kwargs)

어디,

find_next_siblings()find_next_sibling() 메소드는 현재 요소 뒤에 오는 요소의 모든 형제를 반복합니다.

find_previous_siblings()find_previous_sibling() 메소드는 현재 요소 앞에 오는 모든 형제를 반복합니다.

find_all_next()find_next() 메소드는 현재 요소 뒤에 오는 모든 태그와 문자열을 반복합니다.

find_all_previousfind_previous() 메소드는 현재 요소 앞에 오는 모든 태그와 문자열을 반복합니다.

CSS 선택자

가장 일반적으로 사용되는 CSS 선택기를 지원하는 BeautifulSoup 라이브러리. select () 메서드의 도움으로 CSS 선택기를 사용하여 요소를 검색 할 수 있습니다.

다음은 몇 가지 예입니다.

>>> soup.select('title')
[<title>IMDb Top 250 - IMDb</title>, <title>IMDb Top Rated Movies</title>]
>>>
>>> soup.select("p:nth-of-type(1)")
[<p>The Top Rated Movie list only includes theatrical features.</p>, <p> class="imdb-footer__copyright _2-iNNCFskmr4l2OFN2DRsf">© 1990- by IMDb.com, Inc.</p>]
>>> len(soup.select("p:nth-of-type(1)"))
2
>>> len(soup.select("a"))
609
>>> len(soup.select("p"))
2

>>> soup.select("html head title")
[<title>IMDb Top 250 - IMDb</title>, <title>IMDb Top Rated Movies</title>]
>>> soup.select("head > title")
[<title>IMDb Top 250 - IMDb</title>]

#print HTML code of the tenth li elemnet
>>> soup.select("li:nth-of-type(10)")
[<li class="subnav_item_main">
<a href="/search/title?genres=film_noir&sort=user_rating,desc&title_type=feature&num_votes=25000,">Film-Noir
</a> </li>]

BeautifulSoup의 중요한 측면 중 하나는 구문 분석 트리를 검색하는 것이며 요구 사항에 따라 웹 문서를 변경할 수 있습니다. .name, .string 또는 .append () 메소드와 같은 속성을 사용하여 태그의 속성을 변경할 수 있습니다. .new_string () 및 .new_tag () 메서드를 사용하여 기존 태그에 새 태그와 문자열을 추가 할 수 있습니다. HTML 또는 XML 문서를 다양하게 수정하기위한 .insert (), .insert_before () 또는 .insert_after ()와 같은 다른 메소드도 있습니다.

태그 이름 및 속성 변경

스프를 만든 후에는 태그 이름 변경, 속성 수정, 새 속성 추가 및 속성 삭제와 같은 수정을 쉽게 수행 할 수 있습니다.

>>> soup = BeautifulSoup('<b class="bolder">Very Bold</b>')
>>> tag = soup.b

수정 및 새 속성 추가는 다음과 같습니다.

>>> tag.name = 'Blockquote'
>>> tag['class'] = 'Bolder'
>>> tag['id'] = 1.1
>>> tag
<Blockquote class="Bolder" id="1.1">Very Bold</Blockquote>

삭제 속성은 다음과 같습니다-

>>> del tag['class']
>>> tag
<Blockquote id="1.1">Very Bold</Blockquote>
>>> del tag['id']
>>> tag
<Blockquote>Very Bold</Blockquote>

.string 수정

태그의 .string 속성을 쉽게 수정할 수 있습니다.

>>> markup = '<a href="https://www.tutorialspoint.com/index.htm">Must for every <i>Learner>/i<</a>'
>>> Bsoup = BeautifulSoup(markup)
>>> tag = Bsoup.a
>>> tag.string = "My Favourite spot."
>>> tag
<a href="https://www.tutorialspoint.com/index.htm">My Favourite spot.</a>

위에서부터 태그에 다른 태그가 포함되어 있는지 확인할 수 있으며 태그와 모든 내용이 새 데이터로 대체됩니다.

append ()

기존 태그에 새로운 데이터 / 내용을 추가하는 것은 tag.append () 메소드를 사용하는 것입니다. Python 목록의 append () 메서드와 매우 유사합니다.

>>> markup = '<a href="https://www.tutorialspoint.com/index.htm">Must for every <i>Learner</i></a>'
>>> Bsoup = BeautifulSoup(markup)
>>> Bsoup.a.append(" Really Liked it")
>>> Bsoup
<html><body><a href="https://www.tutorialspoint.com/index.htm">Must for every <i>Learner</i> Really Liked it</a></body></html>
>>> Bsoup.a.contents
['Must for every ', <i>Learner</i>, ' Really Liked it']

NavigableString () 및 .new_tag ()

문서에 문자열을 추가하려는 경우 append () 또는 NavigableString () 생성자를 사용하여 쉽게 수행 할 수 있습니다.

>>> soup = BeautifulSoup("<b></b>")
>>> tag = soup.b
>>> tag.append("Start")
>>>
>>> new_string = NavigableString(" Your")
>>> tag.append(new_string)
>>> tag
<b>Start Your</b>
>>> tag.contents
['Start', ' Your']

Note: NavigableString () 함수에 액세스하는 동안 다음과 같이 이름 오류를 발견하면-

NameError : 'NavigableString'이름이 정의되지 않았습니다.

bs4 패키지에서 NavigableString 디렉토리를 가져옵니다.

>>> from bs4 import NavigableString

위의 오류를 해결할 수 있습니다.

기존 태그에 주석을 추가하거나 NavigableString의 다른 하위 클래스를 추가 할 수 있습니다. 생성자를 호출하기 만하면됩니다.

>>> from bs4 import Comment
>>> adding_comment = Comment("Always Learn something Good!")
>>> tag.append(adding_comment)
>>> tag
<b>Start Your<!--Always Learn something Good!--></b>
>>> tag.contents
['Start', ' Your', 'Always Learn something Good!']

완전히 새로운 태그를 추가 (기존 태그에 추가하지 않음)는 Beautifulsoup 내장 메소드 BeautifulSoup.new_tag ()를 사용하여 수행 할 수 있습니다.

>>> soup = BeautifulSoup("<b></b>")
>>> Otag = soup.b
>>>
>>> Newtag = soup.new_tag("a", href="https://www.tutorialspoint.com")
>>> Otag.append(Newtag)
>>> Otag
<b><a href="https://www.tutorialspoint.com"></a></b>

첫 번째 인수 인 태그 이름 만 필요합니다.

끼워 넣다()

파이썬 목록의 .insert () 메서드와 유사하게 tag.insert ()는 새 요소를 삽입하지만 tag.append ()와는 달리 새 요소가 반드시 부모의 내용 끝에있는 것은 아닙니다. 새 요소는 모든 위치에 추가 할 수 있습니다.

>>> markup = '<a href="https://www.djangoproject.com/community/">Django Official website <i>Huge Community base</i></a>'
>>> soup = BeautifulSoup(markup)
>>> tag = soup.a
>>>
>>> tag.insert(1, "Love this framework ")
>>> tag
<a href="https://www.djangoproject.com/community/">Django Official website Love this framework <i>Huge Community base</i></a>
>>> tag.contents
['Django Official website ', 'Love this framework ', <i>Huge Community base</i
>]
>>>

insert_before () 및 insert_after ()

구문 분석 트리에서 무언가 바로 앞에 태그 나 문자열을 삽입하려면 insert_before ()를 사용합니다.

>>> soup = BeautifulSoup("Brave")
>>> tag = soup.new_tag("i")
>>> tag.string = "Be"
>>>
>>> soup.b.string.insert_before(tag)
>>> soup.b
<b><i>Be</i>Brave</b>

유사하게 구문 분석 트리의 항목 바로 뒤에 태그 나 문자열을 삽입하려면 insert_after ()를 사용합니다.

>>> soup.b.i.insert_after(soup.new_string(" Always "))
>>> soup.b
<b><i>Be</i> Always Brave</b>
>>> soup.b.contents
[<i>Be</i>, ' Always ', 'Brave']

맑은()

태그의 내용을 제거하려면 tag.clear ()를 사용하십시오.

>>> markup = '<a href="https://www.tutorialspoint.com/index.htm">For <i>technical & Non-technical&lr;/i> Contents</a>'
>>> soup = BeautifulSoup(markup)
>>> tag = soup.a
>>> tag
<a href="https://www.tutorialspoint.com/index.htm">For <i>technical & Non-technical</i> Contents</a>
>>>
>>> tag.clear()
>>> tag
<a href="https://www.tutorialspoint.com/index.htm"></a>

추출물()

트리에서 태그 또는 문자열을 제거하려면 PageElement.extract ()를 사용하십시오.

>>> markup = '<a href="https://www.tutorialspoint.com/index.htm">For <i&gr;technical & Non-technical</i> Contents</a>'
>>> soup = BeautifulSoup(markup)
>>> a_tag = soup.a
>>>
>>> i_tag = soup.i.extract()
>>>
>>> a_tag
<a href="https://www.tutorialspoint.com/index.htm">For Contents</a>
>>>
>>> i_tag
<i>technical & Non-technical</i>
>>>
>>> print(i_tag.parent)
None

분해 ()

tag.decompose ()는 트리에서 태그를 제거하고 모든 내용을 삭제합니다.

>>> markup = '<a href="https://www.tutorialspoint.com/index.htm">For <i>technical & Non-technical</i> Contents</a>'
>>> soup = BeautifulSoup(markup)
>>> a_tag = soup.a
>>> a_tag
<a href="https://www.tutorialspoint.com/index.htm">For <i>technical & Non-technical</i> Contents</a>
>>>
>>> soup.i.decompose()
>>> a_tag
<a href="https://www.tutorialspoint.com/index.htm">For Contents</a>
>>>

Replace_with ()

이름에서 알 수 있듯이 pageElement.replace_with () 함수는 이전 태그 또는 문자열을 트리의 새 태그 또는 문자열로 대체합니다.

>>> markup = '<a href="https://www.tutorialspoint.com/index.htm">Complete Python <i>Material</i></a>'
>>> soup = BeautifulSoup(markup)
>>> a_tag = soup.a
>>>
>>> new_tag = soup.new_tag("Official_site")
>>> new_tag.string = "https://www.python.org/"
>>> a_tag.i.replace_with(new_tag)
<i>Material</i>
>>>
>>> a_tag
<a href="https://www.tutorialspoint.com/index.htm">Complete Python <Official_site>https://www.python.org/</Official_site></a>

위의 출력에서 ​​replace_with ()가 교체 된 태그 또는 문자열 (예 : "Material")을 반환하므로이를 검사하거나 트리의 다른 부분에 다시 추가 할 수 있습니다.

싸다()

pageElement.wrap ()은 지정한 태그에 요소를 포함하고 새 래퍼를 반환합니다.

>>> soup = BeautifulSoup("<p>tutorialspoint.com</p>")
>>> soup.p.string.wrap(soup.new_tag("b"))
<b>tutorialspoint.com</b>
>>>
>>> soup.p.wrap(soup.new_tag("Div"))
<Div><p><b>tutorialspoint.com</b></p></Div>

풀다()

tag.unwrap ()은 wrap ()과 반대이며 태그를 해당 태그 안에있는 것으로 대체합니다.

>>> soup = BeautifulSoup('<a href="https://www.tutorialspoint.com/">I liked <i>tutorialspoint</i></a>')
>>> a_tag = soup.a
>>>
>>> a_tag.i.unwrap()
<i></i>
>>> a_tag
<a href="https://www.tutorialspoint.com/">I liked tutorialspoint</a>

위에서부터 replace_with ()와 마찬가지로 unwrap ()은 대체 된 태그를 반환합니다.

아래는 더 잘 이해하기 위해 unwrap ()의 또 다른 예입니다.

>>> soup = BeautifulSoup("<p>I <strong>AM</strong> a <i>text</i>.</p>")
>>> soup.i.unwrap()
<i></i>
>>> soup
<html><body><p>I <strong>AM</strong> a text.</p></body></html>

unwrap ()은 마크 업을 제거하는 데 좋습니다.

모든 HTML 또는 XML 문서는 ASCII 또는 UTF-8과 같은 특정 인코딩으로 작성됩니다. 그러나 HTML / XML 문서를 BeautifulSoup에로드하면 유니 코드로 변환됩니다.

>>> markup = "<p>I will display £</p>"
>>> Bsoup = BeautifulSoup(markup)
>>> Bsoup.p
<p>I will display £</p>
>>> Bsoup.p.string
'I will display £'

위의 동작은 BeautifulSoup이 내부적으로 Unicode, Dammit이라는 하위 라이브러리를 사용하여 문서의 인코딩을 감지 한 다음이를 유니 코드로 변환하기 때문입니다.

그러나 항상 유니 코드는 아니지만 Dammit은 올바르게 추측합니다. 인코딩을 추측하기 위해 문서를 바이트 단위로 검색하므로 많은 시간이 걸립니다. 인코딩을 이미 알고있는 경우 BeautifulSoup 생성자에 from_encoding으로 전달하여 시간을 절약하고 실수를 방지 할 수 있습니다.

아래는 BeautifulSoup이 ISO-8859-8 문서를 ISO-8859-7로 잘못 식별 한 예입니다.

>>> markup = b"<h1>\xed\xe5\xec\xf9</h1>"
>>> soup = BeautifulSoup(markup)
>>> soup.h1
<h1>νεμω</h1>
>>> soup.original_encoding
'ISO-8859-7'
>>>

위의 문제를 해결하려면 from_encoding을 사용하여 BeautifulSoup에 전달하십시오.

>>> soup = BeautifulSoup(markup, from_encoding="iso-8859-8")
>>> soup.h1
<h1>ולש </h1>
>>> soup.original_encoding
'iso-8859-8'
>>>

BeautifulSoup 4.4.0에서 추가 된 또 다른 새로운 기능은 exclude_encoding입니다. 올바른 인코딩을 모르지만 Unicode, Dammit이 잘못된 결과를 표시하는 경우 사용할 수 있습니다.

>>> soup = BeautifulSoup(markup, exclude_encodings=["ISO-8859-7"])

출력 인코딩

BeautifulSoup의 출력은 BeautifulSoup에 입력 된 문서에 관계없이 UTF-8 문서입니다. ISO-8859-2 형식의 폴란드어 문자가있는 문서 아래.

html_markup = """
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<HTML>
<HEAD>
<META HTTP-EQUIV="content-type" CONTENT="text/html; charset=iso-8859-2">
</HEAD>
<BODY>
ą ć ę ł ń ó ś ź ż Ą Ć Ę Ł Ń Ó Ś Ź Ż
</BODY>
</HTML>
"""


>>> soup = BeautifulSoup(html_markup)
>>> print(soup.prettify())
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
   <head>
      <meta content="text/html; charset=utf-8" http-equiv="content-type"/>
   </head>
   <body>
      ą ć ę ł ń ó ś ź ż Ą Ć Ę Ł Ń Ó Ś Ź Ż
   </body>
</html>

위의 예에서, <meta> 태그는 BeautifulSoup에서 생성 된 문서가 이제 UTF-8 형식임을 반영하도록 다시 작성되었습니다.

UTF-8로 생성 된 출력을 원하지 않는 경우 prettify ()에서 원하는 인코딩을 할당 할 수 있습니다.

>>> print(soup.prettify("latin-1"))
b'<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">\n<html>\n <head>\n <meta content="text/html; charset=latin-1" http-equiv="content-type"/>\n </head>\n <body>\n ą ć ę ł ń \xf3 ś ź ż Ą Ć Ę Ł Ń \xd3 Ś Ź Ż\n </body>\n</html>\n'

위의 예에서 우리는 완전한 문서를 인코딩했지만, 수프의 특정 요소를 마치 파이썬 문자열 인 것처럼 인코딩 할 수 있습니다.

>>> soup.p.encode("latin-1")
b'<p>0My first paragraph.</p>'
>>> soup.h1.encode("latin-1")
b'<h1>My First Heading</h1>'

선택한 인코딩으로 표현할 수없는 모든 문자는 숫자 XML 엔티티 참조로 변환됩니다. 다음은 그러한 예입니다.

>>> markup = u"<b>\N{SNOWMAN}</b>"
>>> snowman_soup = BeautifulSoup(markup)
>>> tag = snowman_soup.b
>>> print(tag.encode("utf-8"))
b'<b>\xe2\x98\x83</b>'

위의 내용을 "latin-1"또는 "ascii"로 인코딩하려고하면 "☃"가 생성되어 이에 대한 표현이 없음을 나타냅니다.

>>> print (tag.encode("latin-1"))
b'<b>☃</b>'
>>> print (tag.encode("ascii"))
b'<b>☃</b>'

유니 코드, 젠장

Unicode, Dammit은 주로 들어오는 문서가 알 수없는 형식 (주로 외국어)이고 알려진 형식 (유니 코드)으로 인코딩하고 싶을 때 사용되며이 모든 작업을 수행하기 위해 Beautifulsoup이 필요하지 않습니다.

BeautifulSoup 프로젝트의 시작점은 BeautifulSoup 객체입니다. BeautifulSoup 객체는 생성에 사용되는 입력 HTML / XML 문서를 나타냅니다.

Beautiful Soup에 대한 문자열 또는 파일 류 객체를 전달할 수 있습니다. 여기서 파일 (객체)은 컴퓨터 나 웹 페이지에 로컬로 저장됩니다.

가장 일반적인 BeautifulSoup 개체는 다음과 같습니다.

  • Tag
  • NavigableString
  • BeautifulSoup
  • Comment

객체가 같은지 비교

아름다운 수프에 따르면, 두 개의 탐색 가능한 문자열 또는 태그 객체는 동일한 HTML / XML 마크 업을 나타내는 경우 동일합니다.

이제 두 개의 <b> 태그가 "<b> Java </ b>"처럼 보이기 때문에 객체 트리의 다른 부분에 있더라도 동일하게 취급되는 아래 예제를 보겠습니다.

>>> markup = "<p>Learn Python and <b>Java</b> and advanced <b>Java</b>! from Tutorialspoint</p>"
>>> soup = BeautifulSoup(markup, "html.parser")
>>> first_b, second_b = soup.find_all('b')
>>> print(first_b == second_b)
True
>>> print(first_b.previous_element == second_b.previous_element)
False

그러나 두 변수가 동일한 객체를 참조하는지 확인하려면 다음을 사용할 수 있습니다.

>>> print(first_b is second_b)
False

아름다운 수프 개체 복사

태그 또는 NavigableString의 복사본을 만들려면 아래와 같이 copy.copy () 함수를 사용하십시오.

>>> import copy
>>> p_copy = copy.copy(soup.p)
>>> print(p_copy)
<p>Learn Python and <b>Java</b> and advanced <b>Java</b>! from Tutorialspoint</p>
>>>

그러나 두 복사본 (원본과 복사본)에 동일한 마크 업이 포함되어 있지만 두 복사본은 동일한 객체를 나타내지 않습니다.

>>> print(soup.p == p_copy)
True
>>>
>>> print(soup.p is p_copy)
False
>>>

유일한 차이점은 extract ()가 호출 된 것처럼 복사본이 원본 Beautiful Soup 개체 트리에서 완전히 분리된다는 것입니다.

>>> print(p_copy.parent)
None

위의 동작은 동시에 같은 공간을 차지할 수없는 두 개의 서로 다른 태그 개체 때문입니다.

Beautifulsoup4를 사용하여 특정 유형의 정보 (<a> 태그 만)를 추출하려는 여러 상황이 있습니다. Beautifulsoup의 SoupStrainer 클래스를 사용하면 들어오는 문서의 특정 부분 만 구문 분석 할 수 있습니다.

한 가지 방법은 SoupStrainer를 만들고 Parse_only 인수로 Beautifulsoup4 생성자에 전달하는 것입니다.

수프 스트레이너

SoupStrainer는 BeautifulSoup에게 어떤 부분이 추출되는지 알려주고 구문 분석 트리는 이러한 요소로만 구성됩니다. 필요한 정보를 HTML의 특정 부분으로 좁 히면 검색 결과가 빨라집니다.

product = SoupStrainer('div',{'id': 'products_list'})
soup = BeautifulSoup(html,parse_only=product)

위의 코드 줄은 태그 필드 안에있을 수있는 제품 사이트의 제목 만 구문 분석합니다.

마찬가지로 위와 같이 다른 soupStrainer 객체를 사용하여 HTML 태그에서 특정 정보를 구문 분석 할 수 있습니다. 다음은 몇 가지 예입니다.

from bs4 import BeautifulSoup, SoupStrainer

#Only "a" tags
only_a_tags = SoupStrainer("a")

#Will parse only the below mentioned "ids".
parse_only = SoupStrainer(id=["first", "third", "my_unique_id"])
soup = BeautifulSoup(my_document, "html.parser", parse_only=parse_only)

#parse only where string length is less than 10
def is_short_string(string):
   return len(string) < 10
   
only_short_strings =SoupStrainer(string=is_short_string)

오류 처리

BeautifulSoup에서 처리해야하는 두 가지 주요 오류가 있습니다. 이 두 오류는 BeautifulSoup API에서 오류가 발생하기 때문에 스크립트가 아니라 스 니펫 구조에서 발생합니다.

두 가지 주요 오류는 다음과 같습니다.

AttributeError

점 표기법이 현재 HTML 태그에 대한 형제 태그를 찾지 못할 때 발생합니다. 예를 들어, "anchor tag"누락으로 인해이 오류가 발생했을 수 있습니다. cost-key는 트래버스 할 때 오류가 발생하고 앵커 태그가 필요합니다.

KeyError

이 오류는 필수 HTML 태그 속성이 누락 된 경우 발생합니다. 예를 들어 스 니펫에 data-pid 속성이없는 경우 pid 키는 키 오류를 발생시킵니다.

결과를 구문 분석 할 때 위에 나열된 두 가지 오류를 방지하기 위해 해당 결과는 무시되어 잘못된 스 니펫이 데이터베이스에 삽입되지 않도록합니다.

except(AttributeError, KeyError) as er:
pass

진단 ()

BeautifulSoup이 문서 나 HTML에 어떤 일을하는지 이해하는 데 어려움이있을 때마다이를 diagnostic () 함수에 전달하면됩니다. 문서 파일을 diagnostic () 함수에 전달할 때 다른 파서 목록이 문서를 처리하는 방법을 보여줄 수 있습니다.

아래는 diagnostic () 함수의 사용을 보여주는 한 가지 예입니다.

from bs4.diagnose import diagnose

with open("20 Books.html",encoding="utf8") as fp:
   data = fp.read()
   
diagnose(data)

산출

구문 분석 오류

파싱 ​​오류에는 두 가지 주요 유형이 있습니다. BeautifulSoup에 문서를 공급할 때 HTMLParseError와 같은 예외가 발생할 수 있습니다. BeautifulSoup 구문 분석 트리가 구문 분석 문서의 예상 결과와 많이 다르게 보이는 예기치 않은 결과를 얻을 수도 있습니다.

파싱 ​​오류는 BeautifulSoup으로 인해 발생하지 않습니다. BeautifulSoup에는 파서 코드가 포함되어 있지 않기 때문에 우리가 사용하는 외부 파서 (html5lib, lxml) 때문입니다. 위의 구문 분석 오류를 해결하는 한 가지 방법은 다른 구문 분석기를 사용하는 것입니다.

from HTMLParser import HTMLParser

try:
   from HTMLParser import HTMLParseError
except ImportError, e:
   # From python 3.5, HTMLParseError is removed. Since it can never be
   # thrown in 3.5, we can just define our own class as a placeholder.
   class HTMLParseError(Exception):
      pass

Python 내장 HTML 파서는 가장 일반적인 두 가지 구문 분석 오류 인 HTMLParser.HTMLParserError : 잘못된 시작 태그 및 HTMLParser.HTMLParserError : 잘못된 끝 태그를 발생 시키며이를 해결하려면 주로 lxml 또는 html5lib와 같은 다른 구문 분석기를 사용합니다.

또 다른 일반적인 유형의 예기치 않은 동작은 문서에서 알고있는 태그를 찾을 수 없다는 것입니다. 그러나 실행하면 find_all ()이 []를 반환하거나 find ()가 None을 반환합니다.

이것은 파이썬 내장 HTML 파서가 때때로 이해하지 못하는 태그를 건너 뛰기 때문일 수 있습니다.

XML 파서 오류

기본적으로 BeautifulSoup 패키지는 문서를 HTML로 구문 분석하지만, 매우 사용하기 쉽고 beautifulsoup4를 사용하여 매우 우아한 방식으로 형식이 잘못된 XML을 처리합니다.

문서를 XML로 구문 분석하려면 lxml 파서가 있어야하며 Beautifulsoup 생성자에 두 번째 인수로 "xml"을 전달하면됩니다.

soup = BeautifulSoup(markup, "lxml-xml")

또는

soup = BeautifulSoup(markup, "xml")

일반적인 XML 구문 분석 오류는 다음과 같습니다.

AttributeError: 'NoneType' object has no attribute 'attrib'

find () 또는 findall () 함수를 사용하는 동안 일부 요소가 없거나 정의되지 않은 경우에 발생할 수 있습니다.

기타 구문 분석 오류

다음은이 섹션에서 논의 할 다른 파싱 오류 중 일부입니다.

환경 문제

위에서 언급 한 구문 분석 오류 외에도 스크립트가 한 운영 체제에서는 작동하지만 다른 운영 체제에서는 작동하지 않거나 한 가상 환경에서는 작동하지만 다른 가상 환경에서는 작동하지 않거나 작동하지 않는 환경 문제와 같은 다른 구문 분석 문제가 발생할 수 있습니다. 가상 환경 외부. 이러한 모든 문제는 두 환경에서 사용 가능한 파서 라이브러리가 다르기 때문일 수 있습니다.

현재 작업 환경에서 기본 구문 분석기를 확인하거나 확인하는 것이 좋습니다. 현재 작업 환경에서 사용 가능한 현재 기본 파서를 확인하거나 필요한 파서 라이브러리를 BeautifulSoup 생성자에 두 번째 인수로 명시 적으로 전달할 수 있습니다.

대소 문자를 구분하지 않음

HTML 태그와 속성은 대소 문자를 구분하지 않으므로 세 HTML 파서 모두 태그와 속성 이름을 소문자로 변환합니다. 그러나 대 / 소문자가 혼합 된 태그 및 속성을 유지하려면 문서를 XML로 구문 분석하는 것이 좋습니다.

UnicodeEncodeError

아래 코드 세그먼트를 살펴 보겠습니다.

soup = BeautifulSoup(response, "html.parser")
   print (soup)

산출

UnicodeEncodeError: 'charmap' codec can't encode character '\u011f'

위의 문제는 두 가지 주요 상황 때문일 수 있습니다. 콘솔에서 표시하는 방법을 모르는 유니 코드 문자를 인쇄하려고 할 수 있습니다. 둘째, 파일에 쓰기를 시도하고 기본 인코딩에서 지원하지 않는 유니 코드 문자를 전달합니다.

위의 문제를 해결하는 한 가지 방법은 다음과 같이 원하는 결과를 얻기 위해 수프를 만들기 전에 응답 텍스트 / 문자를 인코딩하는 것입니다.

responseTxt = response.text.encode('UTF-8')

KeyError : [attr]

문제의 태그가 attr 속성을 정의하지 않는 경우 tag [ 'attr']에 액세스하여 발생합니다. 가장 일반적인 오류는 "KeyError : 'href'"및 "KeyError : 'class'"입니다. attr이 정의되어 있는지 확실하지 않으면 tag.get ( 'attr')을 사용하십시오.

for item in soup.fetch('a'):
   try:
      if (item['href'].startswith('/') or "tutorialspoint" in item['href']):
      (...)
   except KeyError:
      pass # or some other fallback action

AttributeError

다음과 같이 AttributeError가 발생할 수 있습니다-

AttributeError: 'list' object has no attribute 'find_all'

위의 오류는 주로 find_all ()이 단일 태그 또는 문자열을 반환 할 것으로 예상했기 때문에 발생합니다. 그러나 soup.find_all은 요소의 파이썬 목록을 반환합니다.

해야 할 일은 목록을 반복하고 해당 요소에서 데이터를 포착하는 것입니다.