主頁 > 後端開發 > 豆瓣Top250:Python爬蟲+資料可視化

豆瓣Top250:Python爬蟲+資料可視化

2021-02-12 11:08:39 後端開發

文章目錄

    • 前言
    • 資料爬取
      • 測驗類別庫
        • 1.urllib.request( 獲取網頁HTML源代碼)
        • 2.re(匹配HTML源代碼)
        • 3.bs4(HTML/XML的決議器)
          • 搜索檔案樹
          • CSS選擇器
        • 4.xlwt(決議出的資料保存到Excel)
        • 5.sqlite3(決議出的資料保存到sqlite資料庫)
      • 執行流程
      • 爬取結果
    • 資料可視化
      • Flask框架目錄結構
      • 詞云
      • Echarts圖表

前言

參考B站Python爬蟲基礎5天速成(2021全新合集)Python入門+資料可視化學習制作,以下是我學習程序中的心得體會,由于整個專案太大,部分代碼未做展示,感興趣的伙伴評論留言,可獲取源代碼或相關資源,

資料爬取

from bs4 import BeautifulSoup       # 網頁決議,獲取資料
import re                           # 正則運算式,進行文字匹配
import urllib.request as urlre      # 指定rul,獲取網頁資料
import xlwt                         # 操作Excel
import sqlite3                      # 操作資料庫

測驗類別庫

1.urllib.request( 獲取網頁HTML源代碼)

import urllib.request

url = 'https://movie.douban.com/top250?start='
data = bytes(urllib.parse.urlencode({"":""}),encoding='utf-8')
headers = {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/88.0.4324.96 Safari/537.36' 
}
# 獲取網頁HTML源代碼
urllib.request.urlopen(urllib.request.Request(url=url,data=data,headers=headers,method='POST')).read().decode('utf-8')


# 獲取一個get請求
response = urlre.urlopen('http://www.baidu.com')
# 獲取網頁源代碼
print(response.read().decode('utf-8'))

# response = re.urlopen('http://www.baidu.com')
# 網頁狀態 200
# print(response.status)
# 當我們訪問網頁時瀏覽器會向服務器發送請求頭,里面包含瀏覽器版本等資訊
# 如果我們的爬蟲用get方式直接訪問就會被有反爬機制的服務器識破我們的身份
# 因此我們需要用post方式發送請求,使用代理重新設定headers,并且通過urllib.parse模擬用戶登錄
# print(response.getheaders())
# print(response.getheader('Server'))

# 超時處理
# try:
    # response = re.urlopen('https://httpbin.org/get',timeout=0.5)
    # print(response.read().decode('utf-8'))

# except Exception as e:
#     print(e)
# 獲取一個post請求,模擬用戶登錄時使用
# import urllib.parse
# data = bytes(urllib.parse.urlencode({"":""}),encoding='utf-8')
# response = re.urlopen('https://httpbin.org/post',data=data)
# print(response.read().decode('utf-8'))

url = 'https://movie.douban.com/top250?start='
data = bytes(urllib.parse.urlencode({"":""}),encoding='utf-8')
headers = {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/88.0.4324.96 Safari/537.36'

}
req = re.Request(url=url,data=data,headers=headers,method='POST')
response = re.urlopen(req)
print(response.read().decode('utf-8'))

2.re(匹配HTML源代碼)

# m = re.compile('aa').search('cab')
# m = re.search('aa','dadaa')     # 前面是規則,后面是校驗物件
# <re.Match object; span=(3, 5), match='aa'>

m = re.findall('[a-c]','dkcfjsabaefaf')  # 找到所有匹配的項
# ['c', 'a', 'a', 'a']

m = re.sub('a','A','asdfgdada')     # 找到a,用A替換
# AsdfgdAdA

3.bs4(HTML/XML的決議器)

BeautifulSoup4將復雜HTML檔案轉換成一個復雜的樹形結構,每個節點都是Python物件,所有物件被歸納為4種:

  • Tag 標簽及其內容(查找第一個符合要求的標簽)
    對于 Tag,它有兩個重要的屬性,是 name 和 attrs
  • NavigableString string 標簽內容
  • BeautifulSoup 整個檔案
  • Comment 一個特殊型別的 NavigableString 物件,輸出的內容不包括注釋符號
搜索檔案樹

find_all(name, attrs, recursive, text, *kwargs)

在上面的栗子中我們簡單介紹了find_all的使用,接下來介紹一下find_all的更多用法-過濾器,這些過濾器貫穿整個搜索API,過濾器可以被用在tag的name中,節點的屬性等,

  1. # 兩者是相等的
  2. # t_list = bs.find_all("a") => t_list = bs("a")

傳入引數:

  1. 字串過濾:會查找與字串完全匹配的內容

  2. 正則運算式過濾:如果傳入的是正則運算式,那么BeautifulSoup4會通過search()來匹配內容

  3. 串列:如果傳入一個串列,BeautifulSoup4將會與串列中的任一元素匹配到的節點回傳

  4. 方法:傳入一個方法,根據方法來匹配

CSS選擇器

BeautifulSoup支持發部分的CSS選擇器,在Tag獲取BeautifulSoup物件的.select()方法中傳入字串引數,即可使用CSS選擇器的語法找到Tag

通過標簽名查找

print(bs.select('title'))

屬性查找

print(bs.select('a[class="bri"]'))

更多見beautifulsoup菜鳥教程

4.xlwt(決議出的資料保存到Excel)

import xlwt                         		# 操作Excel
workbook = xlwt.Workbook(encoding='utf-8')  # 創建Excel作業表
worksheet = workbook.add_sheet('sheet1')    # 創建作業簿
worksheet.write(0,0,'hello')                # 在作業簿第一行第一列設定資料
workbook.save('student.xlsx')               # 命名并保存作業表

5.sqlite3(決議出的資料保存到sqlite資料庫)


conn = sqlite3.connect('db.sqlite3')   # 打開或創建資料庫檔案

cursor = conn.cursor()					# 創建游標

# sql = '''
#     create table company(
#         id int primary key not null,
#         name text not null,
#         age int not null,
#         address char(20),
#         salary real
#     );
# '''

# sql = '''
#     insert into company values(1,'張三',10,'重慶',8888);
# '''
result = cursor.execute('select * from company')

for i in result:
    print('id=',i[0])
    print('id=',i[1])
    print('id=',i[2])
    print('id=',i[3])

# result = cursor.fetchall()
# print(result)
# conn.commit()
cursor.close() 
conn.close()

執行流程

在這里插入圖片描述

# 通過urllib.request決議一個指定url內容,回傳網頁HTML原始碼
def askUrl(url):
    headers = {     # 模擬瀏覽器頭部資訊,發送給豆瓣服務器,一般網站有反爬機制,檢測到爬蟲禁止訪問
        'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/88.0.4324.96 Safari/537.36'
        # 用戶代理,避免被發現是爬蟲,告訴豆瓣瀏覽器我們是什么型別的機器(本質上是告訴瀏覽器我們可以接收什么水平的檔案內容)
    }
    req = urlre.Request(url=url, headers=headers)
    try:
        response = urlre.urlopen(req)
        html = response.read().decode('utf-8')
        # print(html)
    except Exception as e:
        print(e)

    return html

# HTML原始碼中找到需要資訊的標簽,用正則模糊匹配
findLink = re.compile(r'<a href="(.*?)"') # 影片詳情鏈接
findImgSrc = re.compile(r'<img.*src="(.*?)"',re.S) # 影片圖片,re.S去掉換行符
......
# 找到子網頁的地址規律,給askUrl決議每個子網頁
# 通過bs4中的BeautifulSoup搜索每個子網頁的檔案樹
# 回圈遍歷每個子網頁,將資訊保存在串列中,最后將串列保存在存放串列的串列中
def getData(baseurl):
    datalist = []
    for i in range(10):   # 獲取頁面資訊十次
        url = baseurl + str(i*25)
        html = askUrl(url)  # 保存獲取到的網頁原始碼
        # 逐一決議資料
        soup = BeautifulSoup(html,'html.parser')
        for item in soup.find_all('div',class_='item'):    # 查找符合要求的字串,形成串列
            # print(item)
            data = []   # 保存一部電影的所有資訊
            item = str(item)

            # 七項資訊,其中titles占兩個位置,陣列中每八位保存一部電影資訊
            # 影片詳情鏈接
            link = re.findall(findLink,item)[0]   # 查找指定字串
            data.append(link)
  			.....
            data.append(bd.strip())     # 去掉前后的空格
            datalist.append(data)       # 將處理好的一部電影資訊放入datalist
            # for i in data:
            #     print(i)
    print('資料爬取成功')
    return datalist
# 保存資料到excel
def saveData(dataList,savePath):
    workbook = xlwt.Workbook(encoding='utf-8')
    worksheet = workbook.add_sheet('sheet1')
    colTitle = ('電影詳情鏈接', '圖片鏈接', '影片中文名', '影片英文名', '影片評分', '評價人數', '影片概況', '相關資訊')
    for i in range(8):
        worksheet.write(0, i, colTitle[i])
    for i in range(250):
        item = dataList[i]
        for j in range(8):
            data = item[j]
            worksheet.write(i + 1, j, data)

    workbook.save(savePath)
# 保存資料到sqlite資料庫
def saveDataBase(dataList,savePath):

    conn = sqlite3.connect(savePath)  # 打開或創建資料庫檔案
    cur = conn.cursor()

    sqlCr = '''
        create table db_top250 if not exists(
            id int primary key autoincrement,
            link text,
            img text,
            cname varchar,
            ename varchar,
            rated int,
            judge int,
            inq text,
            bd text
        );
    '''
    cur.execute(sqlCr)

    for item in dataList:
        for i in range(len(item)):
            if i == 4 or i == 5:
                continue

            item[i] = '"'+item[i]+'"'
        sqlIn = '''
            insert into db_top250(link,img,cname,ename,rated,judge,inq,bd) values(%s)
        ''' % ','.join(item)  # 將,添加在每個item之間
        cur.execute(sqlIn)
        conn.commit()

    cur.close()
    conn.close()

爬取結果

在這里插入圖片描述
在這里插入圖片描述

資料可視化

Flask框架目錄結構

在這里插入圖片描述

詞云

import jieba    # 分詞
from matplotlib import pyplot as plt  # 繪圖,資料可視化
from wordcloud import WordCloud     # 詞云
from PIL import Image               # 圖形處理
import numpy as np                  # 矩陣運算
import sqlite3                      # 資料庫

# 準備詞云所需的文本
con = sqlite3.connect('../../movie.db')
cur = con.cursor()
sql = 'select inq from db_top250'
data = cur.execute(sql)
text = ''
for item in data:
    text = text + item[0]
# print(text)
cur.close()
con.close()

# 分詞
cut = jieba.cut(text)
print(cut)  # 生成器<generator object Tokenizer.cut at 0x0CA06A38>
string = ' '.join(cut)
# print(len(string))  # 5543

img = Image.open('../../tree.jpg')  # 打開遮罩圖片
img_array = np.array(img)           # 將圖片轉化為陣列
wc = WordCloud(
    background_color='white',
    mask=img_array,
    font_path='simfang.ttf'     # 禹衛書法行書簡體.ttf
)
wc.generate_from_text(string)

# 繪制圖片
fig = plt.figure(1)
plt.imshow(wc)
plt.axis('off')  # 坐標軸顯示
# plt.show()

plt.savefig('../../word.jpg',dpi=500)

在這里插入圖片描述

Echarts圖表

Echarts官網

# 需要下載匯入Echarts的js檔案
<script src="../static/js/echarts.min.js"></script>

<script type="text/javascript">

       var myChart = echarts.init(document.getElementById('main-lb'));

       // 指定圖表的配置項和資料
       option = {
      title:{
         text:'豆瓣評分'
      },
      color:['#3398DB'],
      tooltip: {
         trigger: 'axis',
         axisPointer: {            // 坐標軸指示器,坐標軸觸發有效
            type: 'shadow'        // 默認為直線,可選為:'line' | 'shadow'
         }
      },
      grid: {
         left: '3%',
         right: '4%',
         bottom: '3%',
         containLabel: true
      },
      xAxis: [
         {
            type: 'category',
            data: {{ score|tojson }}
                       // ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'],
            /*axisTick: {
               alignWithLabel: true
            }*/
         }
      ],
      yAxis: [
         {
            type: 'value'
         }
      ],
      series: [
         {
            name: 'score',
            type: 'bar',
            barWidth: '60%',
            data: {{count}}
                   // [10, 52, 200, 334, 390, 330, 220]
         }
      ]
   };

       // 使用剛指定的配置項和資料顯示圖表,
       myChart.setOption(option);


   </script>


   {
            type: 'value'
         }
      ],
      series: [
         {
            name: 'score',
            type: 'bar',
            barWidth: '60%',
            data: {{count}}
                   // [10, 52, 200, 334, 390, 330, 220]
         }
      ]
   };

       // 使用剛指定的配置項和資料顯示圖表,
       myChart.setOption(option);


   </script>

在這里插入圖片描述

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/258960.html

標籤:python

上一篇:說說 Python 中的閉包

下一篇:因無聊用python寫了個爬藍奏云的爬蟲(含注釋并包裝為函式)

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more