婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Flask實現異步執行任務

Flask實現異步執行任務

熱門標簽:白銀外呼系統 廣告地圖標注app 激戰2地圖標注 海南400電話如何申請 唐山智能外呼系統一般多少錢 公司電話機器人 哈爾濱ai外呼系統定制 陜西金融外呼系統 騰訊外呼線路

Flask 是 Python 中有名的輕量級同步 web 框架,在一些開發中,可能會遇到需要長時間處理的任務,此時就需要使用異步的方式來實現,讓長時間任務在后臺運行,先將本次請求的響應狀態返回給前端,不讓前端界面「卡頓」,當異步任務處理好后,如果需要返回狀態,再將狀態返回。

怎么實現呢?

使用線程的方式

當要執行耗時任務時,直接開啟一個新的線程來執行任務,這種方式最為簡單快速。

通過 ThreadPoolExecutor 來實現

from flask import Flask
from time import sleep
from concurrent.futures import ThreadPoolExecutor
# DOCS https://docs.python.org/3/library/concurrent.futures.html#concurrent.futures.ThreadPoolExecutor
# 創建線程池執行器
executor = ThreadPoolExecutor(2)
app = Flask(__name__)
@app.route('/jobs')
def run_jobs():
 # 交由線程去執行耗時任務
 executor.submit(long_task, 'hello', 123)
 return 'long task running.'
# 耗時任務
def long_task(arg1, arg2):
 print("args: %s %s!" % (arg1, arg2))
 sleep(5)
 print("Task is done!")
if __name__ == '__main__':
 app.run()

當要執行一些比較簡單的耗時任務時就可以使用這種方式,如發郵件、發短信驗證碼等。

但這種方式有個問題,就是前端無法得知任務執行狀態。

如果想要前端知道,就需要設計一些邏輯,比如將任務執行狀態存儲到 redis 中,通過唯一的任務 id 進行標識,然后再寫一個接口,通過任務 id 去獲取任務的狀態,然后讓前端定時去請求該接口,從而獲得任務狀態信息。

全部自己實現就顯得有些麻煩了,而 Celery 剛好實現了這樣的邏輯,來使用一下。

使用 Celery

為了滿足前端可以獲得任務狀態的需求,可以使用 Celery。

Celery 是實時任務處理與調度的分布式任務隊列,它常用于 web 異步任務、定時任務等,后面單獨寫一篇文章描述 Celery 的架構,這里不深入討論。

現在我想讓前端可以通過一個進度條來判斷后端任務的執行情況。使用 Celery 就很容易實現,首先通過 pip 安裝 Celery 與 redis,之所以要安裝 redis,是因為讓 Celery 選擇 redis 作為「消息代理 / 消息中間件」。

pip install celery
pip install redis

在 Flask 中使用 Celery 其實很簡單,這里先簡單的過一下 Flask 中使用 Celery 的整體流程,然后再去實現具體的項目

1.在 Flask 中初始化 Celery

from flask import Flask
from celery import Celery
app = Flask(__name__)
# 配置
# 配置消息代理的路徑,如果是在遠程服務器上,則配置遠程服務器中redis的URL
app.config['CELERY_BROKER_URL'] = 'redis://localhost:6379/0'
# 要存儲 Celery 任務的狀態或運行結果時就必須要配置
app.config['CELERY_RESULT_BACKEND'] = 'redis://localhost:6379/0'
# 初始化Celery
celery = Celery(app.name, broker=app.config['CELERY_BROKER_URL'])
# 將Flask中的配置直接傳遞給Celery
celery.conf.update(app.config)
 

上述代碼中,通過 Celery 類初始化 celery 對象,傳入的應用名稱與消息代理的連接 URL。

2.通過 celery.task 裝飾器裝飾耗時任務對應的函數

@celery.task
def long_task(arg1, arg2):
 # 耗時任務的邏輯
 return result

3.Flask 中定義接口通過異步的方式執行耗時任務

@app.route('/', methods=['GET', 'POST'])
def index():
 task = long_task.delay(1, 2)
delay () 方法是 applyasync () 方法的快捷方式,applyasync () 參數更多,可以更加細致的控制耗時任務,比如想要 long_task () 在一分鐘后再執行
@app.route('/', methods=['GET', 'POST'])
def index():
 task = long_task.apply_async(args=[1, 2], countdown=60)

delay () 與 apply_async () 會返回一個任務對象,該對象可以獲取任務的狀態與各種相關信息。
通過這 3 步就可以使用 Celery 了。

接著就具體來實現「讓前端可以通過一個進度條來判斷后端任務的執行情況」的需求。

# bind為True,會傳入self給被裝飾的方法
@celery.task(bind=True)
def long_task(self):
 verb = ['Starting up', 'Booting', 'Repairing', 'Loading', 'Checking']
 adjective = ['master', 'radiant', 'silent', 'harmonic', 'fast']
 noun = ['solar array', 'particle reshaper', 'cosmic ray', 'orbiter', 'bit']
 message = ''
 total = random.randint(10, 50)
 for i in range(total):
 if not message or random.random()  0.25:
 # 隨機的獲取一些信息
 message = '{0} {1} {2}...'.format(random.choice(verb),
 random.choice(adjective),
 random.choice(noun))
 # 更新Celery任務狀態
 self.update_state(state='PROGRESS',
 meta={'current': i, 'total': total,
 'status': message})
 time.sleep(1)
 # 返回字典
 return {'current': 100, 'total': 100, 'status': 'Task completed!',
 'result': 42}

上述代碼中,celery.task () 裝飾器使用了 bind=True 參數,這個參數會讓 Celery 將 Celery 本身傳入,可以用于記錄與更新任務狀態。

然后就是一個 for 迭代,迭代的邏輯沒什么意義,就是隨機從 list 中抽取一些詞匯來模擬一些邏輯的運行,為了表示這是耗時邏輯,通過 time.sleep (1) 休眠一秒。

每次獲取一次詞匯,就通過 self.update_state () 更新 Celery 任務的狀態,Celery 包含一些內置狀態,如 SUCCESS、STARTED 等等,這里使用了自定義狀態「PROGRESS」,除了狀態外,還將本次循環的一些信息通過 meta 參數 (元數據) 以字典的形式存儲起來。有了這些數據,前端就可以顯示進度條了。

定義好耗時方法后,再定義一個 Flask 接口方法來調用該耗時方法

@app.route('/longtask', methods=['POST'])
def longtask():
 # 異步調用
 task = long_task.apply_async()
 # 返回 202,與Location頭
 return jsonify({}), 202, {'Location': url_for('taskstatus',
 task_id=task.id)}

簡單而言,前端通過 POST 請求到 /longtask,讓后端開始去執行耗時任務。

返回的狀態碼為 202,202 通常表示一個請求正在進行中,然后還在返回數據包的包頭 (Header) 中添加了 Location 頭信息,前端可以通過讀取數據包中 Header 中的 Location 的信息來獲取任務 id 對應的完整 url。

前端有了任務 id 對應的 url 后,還需要提供一個接口給前端,讓前端可以通過任務 id 去獲取當前時刻任務的具體狀態。

@app.route('/status/task_id>')
def taskstatus(task_id):
 task = long_task.AsyncResult(task_id)
 if task.state == 'PENDING': # 在等待
 response = {
 'state': task.state,
 'current': 0,
 'total': 1,
 'status': 'Pending...'
 }
 elif task.state != 'FAILURE': # 沒有失敗
 response = {
 'state': task.state, # 狀態
 # meta中的數據,通過task.info.get()可以獲得
 'current': task.info.get('current', 0), # 當前循環進度
 'total': task.info.get('total', 1), # 總循環進度
 'status': task.info.get('status', '')
 }
 if 'result' in task.info:
 response['result'] = task.info['result']
 else:
 # 后端執行任務出現了一些問題
 response = {
 'state': task.state,
 'current': 1,
 'total': 1,
 'status': str(task.info), # 報錯的具體異常
 }
 return jsonify(response)

為了可以獲得任務對象中的信息,使用任務 id 初始化 AsyncResult 類,獲得任務對象,然后就可以從任務對象中獲得當前任務的信息。

該方法會返回一個 JSON,其中包含了任務狀態以及 meta 中指定的信息,前端可以利用這些信息構建一個進度條。

如果任務在 PENDING 狀態,表示該任務還沒有開始,在這種狀態下,任務中是沒有什么信息的,這里人為的返回一些數據。如果任務執行失敗,就返回 task.info 中包含的異常信息,此外就是正常執行了,正常執行可以通 task.info 獲得任務中具體的信息。

這樣,后端的邏輯就處理完成了,接著就來實現前端的邏輯,要實現圖形進度條,可以直接使用 nanobar.js,簡單兩句話就可以實現一個進度條,其官網例子如下:

var options = {
 classname: 'my-class',
 id: 'my-id',
 // 進度條要出現的位置
 target: document.getElementById('myDivId')
};
// 初始化進度條對象
var nanobar = new Nanobar( options );
nanobar.go( 30 ); // 30% 進度條
nanobar.go( 76 ); // 76% 進度條
// 100% 進度條,進度條結束
nanobar.go(100);

有了 nanobar.js 就非常簡單了。

先定義一個簡單的 HTML 界面

h2>Long running task with progress updates/h2>
button id="start-bg-job">Start Long Calculation/button>br>br>
div id="progress">/div>

通過 JavaScript 實現對后臺的請求

// 按鈕點擊事件
$(function() {
 $('#start-bg-job').click(start_long_task);
 });
// 請求 longtask 接口
function start_long_task() {
 // 添加元素在html中
 div = $('div class="progress">div>/div>div>0%/div>div>.../div>div> /div>/div>hr>');
 $('#progress').append(div);
 // 創建進度條對象
 var nanobar = new Nanobar({
 bg: '#44f',
 target: div[0].childNodes[0]
 });
 // ajax請求longtask
 $.ajax({
 type: 'POST',
 url: '/longtask',
 // 獲得數據,從響應頭中獲取Location
 success: function(data, status, request) {
 status_url = request.getResponseHeader('Location');
 // 調用 update_progress() 方法更新進度條
 update_progress(status_url, nanobar, div[0]);
 },
 error: function() {
 alert('Unexpected error');
 }
 });
 }
// 更新進度條
function update_progress(status_url, nanobar, status_div) {
 // getJSON()方法是JQuery內置方法,這里向Location中對應的url發起請求,即請求「/status/task_id>」
 $.getJSON(status_url, function(data) {
 // 計算進度
 percent = parseInt(data['current'] * 100 / data['total']);
 // 更新進度條
 nanobar.go(percent);
 // 更新文字
 $(status_div.childNodes[1]).text(percent + '%');
 $(status_div.childNodes[2]).text(data['status']);
 if (data['state'] != 'PENDING'  data['state'] != 'PROGRESS') {
 if ('result' in data) {
 // 展示結果
 $(status_div.childNodes[3]).text('Result: ' + data['result']);
 }
 else {
 // 意料之外的事情發生
 $(status_div.childNodes[3]).text('Result: ' + data['state']);
 }
 }
 else {
 // 2秒后再次運行
 setTimeout(function() {
 update_progress(status_url, nanobar, status_div);
 }, 2000);
 }
 }); 
 } 

可以通過注釋閱讀代碼整體邏輯。

至此,需求實現完了,運行一下。

首先運行 Redis

redis-server

然后運行 celery

celery worker -A app.celery --loglevel=info

最后運行 Flask 項目

python app.py

效果如下:

到此這篇關于Flask實現異步執行任務的文章就介紹到這了,更多相關Flask 異步內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • Python Flask異步發送郵件實現方法解析
  • 深入flask之異步非堵塞實現代碼示例
  • Flask實現異步非阻塞請求功能實例解析

標簽:常德 黔西 益陽 黑龍江 鷹潭 四川 上海 惠州

巨人網絡通訊聲明:本文標題《Flask實現異步執行任務》,本文關鍵詞  Flask,實現,異步,執行任務,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Flask實現異步執行任務》相關的同類信息!
  • 本頁收集關于Flask實現異步執行任務的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    欧美日韩视频在线观看一区二区三区 | 91看片淫黄大片一级在线观看| 国产精品女主播在线观看| 国产99精品国产| 秋霞午夜av一区二区三区| 911精品产国品一二三产区| 成人激情动漫在线观看| 日韩不卡一区二区| 久久成人av少妇免费| 欧美经典三级视频一区二区三区| 99久久免费精品| 成人av电影在线播放| 美女脱光内衣内裤视频久久网站 | 一区二区三区在线视频免费观看| 一本久久a久久免费精品不卡| 免费成人美女在线观看.| 久久久国产精华| 精品国产乱码久久久久久浪潮| 欧美另类久久久品| 666欧美在线视频| 国产成人在线观看| 欧美aaa在线| 亚洲在线观看免费视频| 国产一区二区精品在线观看| 99国产欧美久久久精品| 成人av免费在线播放| 国产精品一区久久久久| 在线精品视频一区二区三四| 欧美日韩国产成人在线91| 精品国产髙清在线看国产毛片| 久久综合九色综合97婷婷女人 | 欧美成人一区二区| 国产精品青草久久| 亚洲色图自拍偷拍美腿丝袜制服诱惑麻豆 | 亚洲精品成a人| 国产在线精品视频| 粉嫩久久99精品久久久久久夜| 色中色一区二区| 日韩一区二区中文字幕| 欧美日韩精品综合在线| 欧美美女bb生活片| 欧美国产日韩亚洲一区| 韩国欧美国产一区| 欧美蜜桃一区二区三区| 欧美裸体一区二区三区| 久久理论电影网| 亚洲一二三区视频在线观看| 国产日产精品1区| 奇米一区二区三区av| 北条麻妃一区二区三区| 日韩亚洲欧美在线| 五月激情综合网| 不卡的av在线| 精品国产免费一区二区三区四区 | 成人免费一区二区三区视频| 亚洲国产欧美日韩另类综合| 久久99热99| 久久久国产精品午夜一区ai换脸| 亚洲成人一区二区| 91激情在线视频| 婷婷中文字幕一区三区| 欧美片网站yy| 舔着乳尖日韩一区| 欧美性猛交xxxxxxxx| 亚洲国产精品二十页| 成人性生交大片免费看中文网站| 欧美xxxxx牲另类人与| 成人开心网精品视频| 日韩欧美国产精品一区| 日韩黄色在线观看| 宅男在线国产精品| 国产aⅴ综合色| 91免费国产在线观看| 亚洲一区二区av在线| 日本精品一级二级| 国产亚洲欧洲997久久综合 | 欧美人牲a欧美精品| 九九国产精品视频| 337p粉嫩大胆色噜噜噜噜亚洲 | zzijzzij亚洲日本少妇熟睡| 午夜av区久久| 亚洲欧洲制服丝袜| 91.com视频| 国产精品三级视频| 91日韩在线专区| 懂色av一区二区在线播放| 精品亚洲成a人| 国产综合色视频| 日本成人在线网站| 日日噜噜夜夜狠狠视频欧美人 | 亚洲欧洲在线观看av| 国产精品素人视频| 久久精品在线观看| 国产亚洲一区二区在线观看| 久久久久久免费网| 欧美激情一区在线观看| 亚洲色图在线播放| 亚洲第四色夜色| 国产剧情一区二区三区| 国产精品综合久久| 91小视频在线| 日韩欧美视频在线| 国产精品久久二区二区| 亚洲成人av一区| 国产露脸91国语对白| 色屁屁一区二区| 26uuu国产电影一区二区| 中文字幕不卡一区| 蜜桃在线一区二区三区| 91蝌蚪国产九色| 精品国产一区二区三区久久久蜜月 | 欧洲色大大久久| 久久久久久99精品| 风间由美中文字幕在线看视频国产欧美 | 日韩精品乱码av一区二区| 91精品国产综合久久久久久| 国产一区在线不卡| 精品国产一区久久| 成人网页在线观看| 午夜影院久久久| 国产精品免费看片| 91.麻豆视频| 色丁香久综合在线久综合在线观看| 国产成人精品免费看| 欧美精品乱人伦久久久久久| 成人免费观看男女羞羞视频| 亚洲v日本v欧美v久久精品| 日本一区免费视频| 欧美视频一区在线| 国产一区中文字幕| 久久国产精品区| youjizz久久| 国产91富婆露脸刺激对白| 日韩午夜电影在线观看| 国产在线国偷精品产拍免费yy| 久久综合九色综合97婷婷女人| 国产福利不卡视频| 亚洲一区二区三区不卡国产欧美| 色一情一乱一乱一91av| 亚洲国产精品久久不卡毛片| 99久久免费精品| 日韩成人午夜精品| 久久亚洲综合色| 日韩欧美国产系列| 亚洲成人自拍偷拍| 91欧美一区二区| 国产一区二区中文字幕| 亚洲精品成a人| 国产精品久久久久久户外露出 | 久久精品日产第一区二区三区高清版| 白白色 亚洲乱淫| 激情综合网av| 图片区小说区国产精品视频| 亚洲国产高清在线| 亚洲精品在线观看网站| 亚洲免费高清视频在线| 欧美日韩精品一区二区在线播放| 国产精品妹子av| 欧美精品色一区二区三区| 一区二区三区四区精品在线视频| 国产精品一卡二卡| 亚洲精品一二三四区| 久久狠狠亚洲综合| 一道本成人在线| 欧美激情资源网| 午夜精品国产更新| 欧美写真视频网站| 中文字幕一区二区三区精华液| 国产最新精品精品你懂的| 欧美浪妇xxxx高跟鞋交| 国产原创一区二区| 亚洲va韩国va欧美va| 欧美精彩视频一区二区三区| 3d成人动漫网站| 欧美最猛黑人xxxxx猛交| 在线一区二区观看| 欧美一级理论片| 欧美一区二区三区成人| 欧美不卡一二三| 亚洲色图在线看| 日本美女视频一区二区| 成人国产精品免费观看视频| 成人av资源下载| 欧美男同性恋视频网站| 国产亚洲欧洲一区高清在线观看| 亚洲四区在线观看| 麻豆91免费看| 色一情一伦一子一伦一区| 欧美一区二区久久久| 亚洲天堂网中文字| 久久av中文字幕片| 久久疯狂做爰流白浆xx| 日韩成人av影视| 欧美私模裸体表演在线观看| 久久香蕉国产线看观看99| 一区二区三区四区蜜桃| 97久久精品人人做人人爽50路| 精品国产91亚洲一区二区三区婷婷| 亚洲欧美日韩小说| 色综合天天性综合|