溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Python中多線程和線程池如何使用

發(fā)布時間:2023-04-18 10:41:40 來源:億速云 閱讀:82 作者:iii 欄目:開發(fā)技術(shù)

本文小編為大家詳細(xì)介紹“Python中多線程和線程池如何使用”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“Python中多線程和線程池如何使用”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學(xué)習(xí)新知識吧。

一、多線程

多線程是指在同一進(jìn)程中,有多個線程同時執(zhí)行不同的任務(wù)。Python中的多線程是通過threading模塊來實現(xiàn)的。下面是一個簡單的多線程示例:

import threading

def task(num):
    print('Task %d is running.' % num)

if __name__ == '__main__':
    for i in range(5):
        t = threading.Thread(target=task, args=(i,))
        t.start()

上述代碼中,我們定義了一個task函數(shù),它接受一個參數(shù)num,用于標(biāo)識任務(wù)。在主程序中,我們創(chuàng)建了5個線程,每個線程都執(zhí)行task函數(shù),并傳入不同的參數(shù)。通過start()方法啟動線程。運行上述代碼,可以看到輸出結(jié)果類似于下面這樣:

Task 0 is running.
Task 1 is running.
Task 2 is running.
Task 3 is running.
Task 4 is running.

由于多線程是并發(fā)執(zhí)行的,因此輸出結(jié)果的順序可能會有所不同。

二、線程池

線程池是一種管理多線程的機(jī)制,它可以預(yù)先創(chuàng)建一定數(shù)量的線程,并將任務(wù)分配給這些線程執(zhí)行。Python中的線程池是通過ThreadPoolExecutor類來實現(xiàn)的。下面是一個簡單的線程池示例:

import concurrent.futures

def task(num):
    print('Task %d is running.' % num)

if __name__ == '__main__':
    with concurrent.futures.ThreadPoolExecutor(max_workers=3) as executor:
        for i in range(5):
            executor.submit(task, i)

上述代碼中,我們使用了with語句創(chuàng)建了一個ThreadPoolExecutor對象,其中max_workers參數(shù)指定了線程池中最大的線程數(shù)量。在主程序中,我們創(chuàng)建了5個任務(wù),每個任務(wù)都通過executor.submit()方法提交給線程池執(zhí)行。運行上述代碼,可以看到輸出結(jié)果類似于下面這樣:

Task 0 is running.
Task 1 is running.
Task 2 is running.
Task 3 is running.
Task 4 is running.

由于線程池中最大的線程數(shù)量為3,因此只有3個任務(wù)可以同時執(zhí)行,其他任務(wù)需要等待線程池中的線程空閑后再執(zhí)行。

三、使用案例

下面是一個實際的案例,展示了如何使用多線程和線程池來加速數(shù)據(jù)處理過程。假設(shè)我們有一個包含1000個元素的列表,需要對每個元素進(jìn)行某種運算,并將結(jié)果保存到另一個列表中。我們可以使用單線程的方式來實現(xiàn):

def process(data):
    result = []
    for item in data:
        result.append(item * 2)
    return result

if __name__ == '__main__':
    data = list(range(1000))
    result = process(data)
    print(result)

上述代碼中,我們定義了一個process函數(shù),它接受一個列表作為參數(shù),對列表中的每個元素進(jìn)行運算,并將結(jié)果保存到另一個列表中。在主程序中,我們創(chuàng)建了一個包含1000個元素的列表,并將其傳遞給process函數(shù)。運行上述代碼,可以看到輸出結(jié)果類似于下面這樣:

[0, 2, 4, 6, 8, ..., 1996, 1998]

Python中的多線程和線程池可以提高爬蟲的效率,本文將介紹一個爬取豆瓣電影Top250的案例,并通過多線程和線程池優(yōu)化爬取過程。

1.單線程爬取

首先,我們先來看一下單線程爬取的代碼:

# -*- coding: utf-8 -*-
import requests
from bs4 import BeautifulSoup


def get_html(url):
    try:
        response = requests.get(url)
        if response.status_code == 200:
            return response.text
        else:
            return None
    except Exception as e:
        print(e)


def parse_html(html):
    soup = BeautifulSoup(html, 'lxml')
    movie_list = soup.find(class_='grid_view').find_all('li')

    for movie in movie_list:
        title = movie.find(class_='title').string
        rating = movie.find(class_='rating_num').string
        print(title, rating)


def main():
    url = 'https://movie.douban.com/top250'
    html = get_html(url)
    parse_html(html)


if __name__ == '__main__':
    main()

這是一個簡單的爬取豆瓣電影Top250的代碼,首先通過requests庫獲取網(wǎng)頁的HTML代碼,然后使用BeautifulSoup庫解析HTML代碼,獲取電影名稱和評分。

但是,這種單線程爬取的方式效率較低,因為在獲取HTML代碼的時候需要等待響應(yīng),而在等待響應(yīng)的過程中CPU會空閑,無法充分利用計算機(jī)的性能。

2.多線程爬取

接下來,我們通過多線程的方式來優(yōu)化爬取過程。首先,我們需要導(dǎo)入Python中的threading庫:

import threading

然后,我們將獲取HTML代碼的代碼放在一個函數(shù)中,并將其作為一個線程來運行:

def get_html(url):
    try:
        response = requests.get(url)
        if response.status_code == 200:
            return response.text
        else:
            return None
    except Exception as e:
        print(e)


class GetHtmlThread(threading.Thread):
    def __init__(self, url):
        threading.Thread.__init__(self)
        self.url = url

    def run(self):
        html = get_html(self.url)
        parse_html(html)

在上面的代碼中,我們首先定義了一個GetHtmlThread類,繼承自threading.Thread類,然后在類的構(gòu)造函數(shù)中傳入需要爬取的URL。在run方法中,我們調(diào)用get_html函數(shù)獲取HTML代碼,并將其傳入parse_html函數(shù)中進(jìn)行解析。

接下來,我們通過循環(huán)創(chuàng)建多個線程來進(jìn)行爬?。?/p>

def main():
    urls = ['https://movie.douban.com/top250?start={}'.format(i) for i in range(0, 250, 25)]
    threads = []

    for url in urls:
        thread = GetHtmlThread(url)
        thread.start()
        threads.append(thread)

    for thread in threads:
        thread.join()

在上面的代碼中,我們首先定義了一個urls列表,包含了所有需要爬取的URL。然后通過循環(huán)創(chuàng)建多個GetHtmlThread線程,并將其加入到threads列表中。最后,通過循環(huán)調(diào)用join方法等待所有線程執(zhí)行完畢。

通過多線程的方式,我們可以充分利用計算機(jī)的性能,提高爬取效率。

3.線程池爬取

在多線程的方式中,我們需要手動創(chuàng)建和管理線程,這樣會增加代碼的復(fù)雜度。因此,我們可以使用Python中的線程池來進(jìn)行優(yōu)化。

首先,我們需要導(dǎo)入Python中的concurrent.futures庫:

import concurrent.futures

然后,我們將獲取HTML代碼的代碼放在一個函數(shù)中,并將其作為一個任務(wù)來提交給線程池:

def get_html(url):
    try:
        response = requests.get(url)
        if response.status_code == 200:
            return response.text
        else:
            return None
    except Exception as e:
        print(e)


def parse_html(html):
    soup = BeautifulSoup(html, 'lxml')
    movie_list = soup.find(class_='grid_view').find_all('li')

    for movie in movie_list:
        title = movie.find(class_='title').string
        rating = movie.find(class_='rating_num').string
        print(title, rating)


def main():
    urls = ['https://movie.douban.com/top250?start={}'.format(i) for i in range(0, 250, 25)]
    with concurrent.futures.ThreadPoolExecutor(max_workers=5) as executor:
        futures = [executor.submit(get_html, url) for url in urls]

    for future in concurrent.futures.as_completed(futures):
        html = future.result()
        parse_html(html)

在上面的代碼中,我們首先定義了一個urls列表,包含了所有需要爬取的URL。然后通過with語句創(chuàng)建一個線程池,并設(shè)置最大線程數(shù)為5。接下來,我們通過循環(huán)將每個URL提交給線程池,并將返回的Future對象加入到futures列表中。最后,通過concurrent.futures.as_completed函數(shù)來等待所有任務(wù)執(zhí)行完畢,并獲取返回值進(jìn)行解析。

通過線程池的方式,我們可以更加簡潔地實現(xiàn)多線程爬取,并且可以更加靈活地控制線程的數(shù)量,避免線程過多導(dǎo)致系統(tǒng)負(fù)載過高的問題。

讀到這里,這篇“Python中多線程和線程池如何使用”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領(lǐng)會,如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI