python 3.x 爬蟲基礎---正則表達式

python 3.x 爬蟲基礎

python 3.x 爬蟲基礎---http headers詳解html

python 3.x 爬蟲基礎---Urllib詳解python

python 3.x 爬蟲基礎---Requersts,BeautifulSoup4(bs4)正則表達式

python 3.x 爬蟲基礎---正則表達式json

前言

  正則表達式是對字符串的一種邏輯公式,用事先定義好的一些特定字符、及這些特定字符的組合,組成一個「規則的字符串」,此字符串用來表示對字符串的一種「過濾」邏輯。正在在不少開發語言中都存在,而非python獨有。對其知識點進行總結後,會寫一個demo。多線程

1.正則表達式

  python是自1.5開始引進re模塊進行處理正則的。我先把正則的匹配規則總結一下,再總結re模塊相應的方法。app

1.1匹配規則

語法 解釋 表達式 成功匹配對象
通常字符 匹配自身相對應的字符 abc abc
. 匹配除換行符(\n)之外的任意字符 a.c abc
\ 轉義字符,能夠改變原字符的意思 a.c a.c
\d 匹配數字:0~9 \dabc 1abc
\w 匹配單詞字符,a~z;A~Z;0~9 \w\w\w oX2
\s 匹配空格字符(\t,\n,\r,\f,\v) a\sc a c
\D 匹配非數字字符 \Dabc aabc
\W 匹配非單詞字符 a\Wc a c
\S 匹配非空格字符 \S\Sc 1bc
[] 字符集,對應位置上能夠是字符集裏的任意字符 a[def]c aec
[^] 對字符集當中的內容進行取反 a[^def]c a2c
[a-z] 指定一個範圍字符集 a[A-Z]c aBc
* 容許前一個字符能夠出現0次或者無限次 a*b aaab或b
+ 前一個字符至少出現1次 a+b aaab或ab
? 前一個字符只能出現一次或者不出現 a?b ab或b
{m} 容許前一個字符只能出現m次 a{3}b aaab
{m,n} 容許前一個字符至少出現m次,最多出現n次(若是不寫n,則表明至少出現m次) a{3,5}b和a{3,} aaaab和aaaaaab
^ 匹配字符串的開始,多行內容時匹配每一行的開始 ^abc abc
$ 匹配字符串的結尾,多行內容時匹配每一行的結尾 abc& abc
\A 匹配字符串開始位置,忽略多行模式 \Aabc abc
\Z 匹配字符串結束位置,忽略多行模式 abc\Z abc
\b 匹配位於單詞開始或結束位置的空字符串 hello \bworld hello world
\B 匹配不位於單詞開始或結束位置的空字符串 he\Bllo hello
| 表示左右表達式任意知足一種便可 abc|cba abc或cba
(…) 將被括起來的表達式做爲一個分組,可使用索引單獨取出 (abc)d abcd
(?P<name>…) 爲該分組起一個名字,能夠用索引或名字去除該分組 (?P<id>abc)d abcd
\number 引用索引爲number中的內容 (abc)d\1 abcdabc
(?P=name) 引用該name分組中的內容 (?P<id>abc)d(?P=id) abcdabc
(?:…) 分組的不捕獲模式,計算索引時會跳過這個分組 (?:a)b(c)d\1 abcdc
(?iLmsux) 分組中能夠設置模式,iLmsux之中的每一個字符表明一個模式 (?i)abc Abc
(?#…) 註釋,#後面的內容會被忽略 ab(?#註釋)123 ab123
(?=…) 順序確定環視,表示所在位置右側可以匹配括號內正則 a(?=\d) a1最後的結果獲得a
(?!…) 順序否認環視,表示所在位置右側不能匹配括號內正則 a(?!\w) a c最後的結果獲得a
(?<=…) 逆序確定環視,表示所在位置左側可以匹配括號內正則 1(?<=\w)a 1a
(?<!…) 逆序否認環視,表示所在位置左側不能匹配括號內正則 1 (?<!\w)a 1 a
(?(id/name)yes|no) 若是前面的索引爲id或者名字爲name的分組匹配成功則匹配yes區域的表達式,不然匹配no區域的表達式,no能夠省略 (\d)(?(1)\d|a) 32

  上面表格中(?iLmsux)這裏的」i」, 「L」, 「m」, 「s」, 「u」, 「x」,它們不匹配任何字串,而對應re模塊中(re.S|re.S):函數

I:re.I# 忽略大小寫
L:re.L# 字符集本地化,爲了支持多語言版本的字符集使用環境
U :re.U# 使用\w,\W,\b,\B這些元字符時將按照UNICODE定義的屬性
M:re.M # 多行模式,改變 ^ 和 $ 的行爲
S:re.S  # '.' 的匹配不受限制,包括換行符
X:re.X # 冗餘模式,能夠忽略正則表達式中的空白和#號的註釋

對於一個特殊字符在正則表達式中是不能正常識別的,若是接觸過其餘語言咱們就這到有一個叫作轉移字符的東西的存在,在特殊字符前加用反斜槓接口。好比\n換行\\爲反斜槓,在這再也不累述。下面來介紹一下re這個模塊。post

1.2.re模塊

 此模塊主要方法以下編碼

re.match()#嘗試從字符串的起始位置匹配一個模式(pattern),若是不是起始位置匹配成功的話,match()就返回None
re.search()#函數會在字符串內查找模式匹配,只要找到第一個匹配而後返回,若是字符串沒有匹配,則返回None。
re.findall()#遍歷匹配,能夠獲取字符串中全部匹配的字符串,返回一個列表。
re.compile()#編譯正則表達式模式,返回一個對象的模式。(能夠把那些經常使用的正則表達式編譯成正則表達式對象,這樣能夠提升一點效率。)
re.sub()#使用re替換string中每個匹配的子串後返回替換後的字符串。
re.subn()#返回替換次數
re.split()#按照可以匹配的子串將string分割後返回列表。

1.2.1.re.match()

方法: re.match(pattern, string, flags=0)#pattern:正則表達式(或者正則表達式對象)string:要匹配的字符串flags:修飾符url

  先看一個最簡單的用法

import re
content ='Hello 123 4567 wangyanling REDome'
print(len(content))
result = re.match('^Hello\s\d\d\d\s\d{4}\s\w{10}.*Dome$', content)
print(result)
print(result.group())
print(result.span())

  結果:

匹配規則就不在累述,以上須要注意的是

(1).group()表示的是返回正則匹配的結果

(2).span()表示返回正則匹配的範圍

使用:

以上咱們已經知道re.matcha()的具體方法,那麼接下我來看一下具體使用,對此咱們要理解如下幾種匹配的感念。

  1.泛匹配(.*):匹配全部字符

import re
content ='Hello 123 4567 wangyanling REDome'
result = re.match('^Hello.*Dome$', content)
print(result)
print(result.group())
print(result.span())

它的結果是和上面的輸出結果徹底同樣的。

  2.目標匹配(()):將須要的字符匹配出來

import re
content ='Hello 123 4567 wangyanling REDome'
result = re.match('^Hello\s\d\d(\d)\s\d{4}\s\w{10}.*Dome$', content)
print(result)
print(result.group(1))
import re
content ='Hello 123 4567 wangyanling REDome'
result = re.match('^Hello\s(\d+)\s\d{4}\s\w{10}.*Dome$', content)
print(result)
print(result.group(1))

結果

以上能夠看出:

(1)()匹配括號內的表達式,也表示一個組
(2)+ 匹配1個或多個的表達式
* 匹配0個或多個的表達式
(3).group(1)—輸出第一個帶有()的目標

   3.貪婪匹配(.*()):匹配儘量少的的結果

import re
content ='Hello 123 4567 wangyanling REDome'
result = re.match('^H.*(\d+).*Dome$', content)
print(result)
print(result.group(1))

結果

     4.貪婪匹配(.*?()):匹配儘量多的結果

import re
content ='Hello 123 4567 wangyanling REDome'
result = re.match('^H.*?(\d+).*?Dome$', content)
print(result)
print(result.group(1))

結果

以上3,4兩個匹配方式請儘可能採用非貪婪匹配

    5.其餘

換行:

import re
content ='''Hello 123 4567 
         wangyanling REDome'''

result = re.match('^H.*?(\d+).*?Dome$', content,re.S)#re.S
print(result.group(1))
result = re.match('^H.*?(\d+).*?Dome$', content)
print(result.group(1))

結果:

轉義字符:

import re
content = 'price is $5.00'
result = re.match('price is $5.00', content)
print(result)
result = re.match('price is \$5\.00', content)
print(result)

結果:

 其中re.I使匹配對大小不敏感,re.S匹配包括換行符在內的全部字符,\進行處理轉義字符。匹配規則中有詳細介紹。

1.2.2.re.search()

方法: 

re.search(pattern, string, flags=0)#pattern:正則表達式(或者正則表達式對象)string:要匹配的字符串flags:修飾符
    #re.match()和re.search()用法相似惟一的區別在於re.match()從字符串頭開始匹配,若頭匹配不成功,則返回None    

對比一下與match()

import re
content ='Hello 123 4567 wangyanling REDome'
result = re.match('(\d+)\s\d{4}\s\w{10}.*Dome$', content)
print(result)#從開頭開始查找,不能匹配返回None
result = re.search('(\d+)\s\d{4}\s\w{10}.*Dome$', content)
print(result)
print(result.group())

 結果:

能夠看出兩個使用基本一致,search從頭開始匹配,若是匹配不到就返回none.

1.2.3.re.findall()

方法: re.finditer(pattern, string, flags=0)#pattern:正則表達式(或者正則表達式對象)string:要匹配的字符串flags:修飾符 

與re.search()相似區別在於re.findall()搜索string,返回一個順序訪問每個匹配結果(Match對象)的迭代器。找到 RE 匹配的全部子串,並把它們做爲一個迭代器返回。

import re

html = '''
  <div>
  <li><a href="" singer="魯迅">吶喊</a></li>
  <li><a href="#" singer="賈平凹">廢都</a></li>
  <li class="active"><a href="#" singer="路遙">平凡世界</a></li>
  <span class="rightSpan">謝謝支持</span>
  </div>
'''
regex_4='<a.*?>(.*?)</a>'
results=re.findall(regex_4,html,re.S)
print(results)
for result in results:
    print(result)

結果:

1.2.4.re.compile()

編譯正則表達式模式,返回一個對象的模式。

方法: re.compile(pattern,flags=0)#pattern:正則表達式(或者正則表達式對象);flags:修飾符 

看一個demo

import re
content ='Hello 123 4567 wangyanling REDome wangyanling 那小子很帥'
rr = re.compile(r'\w*wang\w*')
result =rr.findall(content)
print(result)

結果:

咱們能夠看出compile 咱們能夠把它理解爲封裝了一個公用的正則,相似於方法,而後功用。

1.2.5.其餘

re.sub 替換字符

方法: re.sub(pattern, repl, string, count=0, flags=0)#pattern:正則表達式(或者正則表達式對象)repl:替換的字符串string:要匹配的字符串count:要替換的個數flags:修飾符 

re.subn 替換次數

方法: re.subn(pattern, repl, string, count=0, flags=0)#pattern:正則表達式(或者正則表達式對象)repl:替換的字符串string:要匹配的字符串count:要替換的個數flags:修飾符 

re.split()分隔字符

方法

re.split(pattern, string,[maxsplit])#正則表達式(或者正則表達式對象)string:要匹配的字符串;maxsplit:用於指定最大分割次數,不指定將所有分割

2.案例:爬取貓眼信息,寫入txt,csv,下載圖片

2.1.獲取單頁面信息

def get_one_page(html):
    pattern= re.compile('<dd>.*?board-index.*?>(\d+)</i>.*?data-src="(.*?)".*?name"><a.*?>(.*?)</a>.*?star">(.*?)</p>.*?releasetime'
                         + '.*?>(.*?)</p>.*?score.*?integer">(.*?)</i>.*?>(.*?)</i>.*?</dd>',re.S)#這裏就用到了咱們上述提到的一些知識點,非貪婪匹配,對象匹配,修飾符
    items = re.findall(pattern,html)
    for item in  items:
        yield {
            'rank' :item[0],
            'img':  item[1],
            'title':item[2],
            'actor':item[3].strip()[3:] if len(item[3])>3 else '',  
            'time' :item[4].strip()[5:] if len(item[4])>5 else '',
            'score':item[5] + item[6]
        }

對於上面的信息咱們能夠看出是存到一個對象中那麼接下來咱們應該把它們存到文件當中去。

2.2.保存文件

我寫了兩種方式保存到txt和csv這些在python都有涉及,不懂得能夠去翻看一下。

2.2.1.保存到txt

def write_txtfile(content):
    with open("Maoyan.txt",'a',encoding='utf-8') as f:
        #要引入json,利用json.dumps()方法將字典序列化,存入中文要把ensure_ascii編碼方式關掉
        f.write(json.dumps(content,ensure_ascii=False) + "\n")
        f.close()

結果:

以上看到並不是按順序排列由於我用的是多線程。

2.2.2.保存到csv

def write_csvRows(content,fieldnames):
    '''寫入csv文件內容'''
    with open("Maoyao.csv",'a',encoding='gb18030',newline='') as f:
        #將字段名傳給Dictwriter來初始化一個字典寫入對象
        writer = csv.DictWriter(f,fieldnames=fieldnames)
        #調用writeheader方法寫入字段名
        writer.writerows(content)
        f.close()

結果:

那麼還有一部就是咱們要把圖片下載下來。

2.2.3.下載圖片

def download_img(title,url):
   r=requests.get(url)
   with open(title+".jpg",'wb') as f:
        f.write(r.content)

2.3.總體代碼

這裏面又到了多線程在這不在敘述後面會有相關介紹。這個demo僅作一案例,主要是對正則能有個認知。上面寫的知識點有不足的地方望你們多多指教。

#抓取貓眼電影TOP100榜
from multiprocessing import Pool
from requests.exceptions import RequestException
import requests
import json
import time
import csv
import re
def get_one_page(url):
    '''獲取單頁源碼'''
    try:
        headers = {
            "User-Agent":"Mozilla/5.0(WindowsNT6.3;Win64;x64)AppleWebKit/537.36(KHTML,likeGecko)Chrome/68.0.3440.106Safari/537.36"
        }
        res = requests.get(url, headers=headers)
        # 判斷響應是否成功,若成功打印響應內容,不然返回None
        if res.status_code == 200:
            return res.text
        return None
    except RequestException:
        return None
def parse_one_page(html):
    '''解析單頁源碼'''
    pattern = re.compile('<dd>.*?board-index.*?>(\d+)</i>.*?data-src="(.*?)".*?name"><a.*?>(.*?)</a>.*?star">(.*?)</p>.*?releasetime'
                         + '.*?>(.*?)</p>.*?score.*?integer">(.*?)</i>.*?>(.*?)</i>.*?</dd>',re.S)
    items = re.findall(pattern,html)
    #採用遍歷的方式提取信息
    for item in  items:
        yield {
            'rank' :item[0],
            'img':  item[1],
            'title':item[2],
            'actor':item[3].strip()[3:] if len(item[3])>3 else '',  #判斷是否大於3個字符
            'time' :item[4].strip()[5:] if len(item[4])>5 else '',
            'score':item[5] + item[6]
        }

def write_txtfile(content):
    with open("Maoyan.txt",'a',encoding='utf-8') as f:
        #要引入json,利用json.dumps()方法將字典序列化,存入中文要把ensure_ascii編碼方式關掉
        f.write(json.dumps(content,ensure_ascii=False) + "\n")
        f.close()
def write_csvRows(content,fieldnames):
    '''寫入csv文件內容'''
    with open("Maoyao.csv",'a',encoding='gb18030',newline='') as f:
        #將字段名傳給Dictwriter來初始化一個字典寫入對象
        writer = csv.DictWriter(f,fieldnames=fieldnames)
        #調用writeheader方法寫入字段名
        #writer.writeheader()            ###這裏寫入字段的話會形成在抓取多個時重複.
        writer.writerows(content)
        f.close()
def download_img(title,url):
   r=requests.get(url)
   with open(title+".jpg",'wb') as f:
        f.write(r.content)
def main(offset):
    fieldnames = ["rank","img", "title", "actor", "time", "score"]
    url = "http://maoyan.com/board/4?offset={0}".format(offset)
    html = get_one_page(url)
    rows = []
    for item in parse_one_page(html):
        #download_img(item['rank']+item['title'],item['img'])
        write_txtfile(item)
        rows.append(item)
    write_csvRows(rows,fieldnames)

if __name__ == '__main__':
    pool = Pool()
    #map方法會把每一個元素當作函數的參數,建立一個個進程,在進程池中運行.
    pool.map(main,[i*10 for i in range(10)])
相關文章
相關標籤/搜索