Mongodb可以使用BSON格式來保存大小不超過16M的二進制文件,很適合用來存放web中的圖片資源,記錄一下操作BSON格式的方法
本地文件寫入Pymongo/Pymongo數據寫回本地文件
"""
@File : file_demo.py
@Description : 本地文件寫入Pymongo/Pymongo數據寫回本地文件
@Time : 2020/4/7 23:20
@Author : DexterLien
@Email : [email protected]
@Software : PyCharm
"""
import os
from pymongo import MongoClient
# 連接帶有密碼認證的Mongodb,格式如下:
# mongodb://用戶名:密碼@服務器地址/數據庫名
con = MongoClient('mongodb://username:[email protected]:8017/dexter')
# 使用名爲dexter的數據庫
db = con['dexter']
# 使用名爲users的collection
users = db['users']
# 本地文件以BSON二進制寫入數據庫集合
fpath = os.path.join(os.getcwd(), 'pic.png')
with open(fpath, 'rb') as file:
users.insert_one({
'name': 'pic.png',
'data': file.read()
})
# 數據庫中的BSON數據寫回文件
ret = users.find_one({'name': 'pic.png'})
with open(os.path.join(os.getcwd(), 'new.png'), 'wb') as file:
file.write(ret.get('data'))
POST上傳的圖片附件寫入Mongodb與返回
用Flask隨便擼了一個demo,記錄一下主要的思路
"""
@File : post_pic.py
@Description : POST上傳的圖片附件寫入Mongodb與返回
@Time : 2020/4/7 23:59
@Author : DexterLien
@Email : [email protected]
@Software : PyCharm
"""
from flask import Flask, request, render_template, jsonify, make_response
from pymongo import MongoClient
from werkzeug.datastructures import FileStorage
app = Flask(__name__)
app.config['JSON_AS_ASCII'] = False
def save_to_mongo(file: FileStorage):
"""
將post請求中的文件對象保存至Mongodb,主要是通過file.stream.read()方法直接將二進制數據讀出來,給Mongodb插入用
:param file:
:return:
"""
con = MongoClient('mongodb://username:[email protected]:8017/dexter')
db = con['dexter']
users = db['users']
users.insert_one({
'name': file.filename,
'data': file.stream.read()
})
@app.route('/', methods=['GET', 'POST'])
def index():
if request.method == 'GET':
return render_template('index.html')
if request.method == 'POST':
file = request.files.get('pic')
save_to_mongo(file)
return jsonify({
'code': 0,
'msg': '上傳成功'
})
@app.route('/pic/<filename>', methods=['GET'])
def pic(filename):
con = MongoClient('mongodb://username:[email protected]:8017/dexter')
db = con['dexter']
users = db['users']
data = users.find_one({'name': filename})['data']
# 使用Mongodb查詢到的BSON數據生成response響應,後面通過設置不同的headers實現不同響應行爲
response = make_response(data)
# 在瀏覽器中直接顯示圖片
response.headers.set('Content-Type', 'image/png')
# 以文件下載方式返回
# response.headers.set(
# 'Content-Disposition', 'attachement', filename=f'{filename}'
# )
return response
if __name__ == '__main__':
app.run(debug=True)
上傳操作前端頁面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
{# 注意:Flask中需要指定form的enctype才能將文件封裝發送到post請求中 #}
<form method="post" enctype=multipart/form-data>
<input name="pic" type="file">
<button type="submit">上傳</button>
</form>
</body>
</html>
使用Flask直接將圖片二進制數據傳給jinja2模板渲染
主要是使用base64.b64encode()
先將Mongodb中的二進制數據進行Base64編碼,然後再對其使用utf-8
解碼,這樣就可以得到圖像的Base64
類型的字符串數據
import base64
# 省略其他代碼
@bp.route('/users')
def users_view():
_users = users.find({})
data = [u for u in _users]
for d in data:
d['mgcz'] = base64.b64encode(d['mgcz']).decode('utf-8')
return render_template('users.html', users=data)
jinja2
模板中使用<img src="data:;base64,{{ user.mgcz | safe }}">
這樣格式的標籤來渲染Base64
格式字符串數據的圖片
<div>
<ul>
{% for user in users %}
<li>
{{ user.xm }}
<img src="data:;base64,{{ user.mgcz | safe }}">
</li>
{% endfor %}
</ul>
</div>
與傳統文件方式存儲圖片佔用空間對比
做了一個簡單的測試:
傳統文件方式存放的圖片new.webp
文件實際大小:
使用BSON類型向MongoDB中寫入該圖片文件的前後服務器上數據庫文件大小對比(單位byte字節)
lpwm@python-server:~/mongodb$ sudo du -sb
336945978 .
lpwm@python-server:~/mongodb$ sudo du -sb
336947156
相減後服務器上MongoDB存儲這個圖片數據只用了1,178
個字節,比傳統文件方式佔用的60,362
字節大小減少了98%,簡直太叼了有木有!!!感覺這樣對比是不是有點不準確呢?