我需要将 PNG 文件作为 blob/二进制存储在数据库中,然后能够检索并显示它们。
这是我的模型,它有一个二进制字段来存储图像:
class ImageFile(models.Model):
file = models.BinaryField(editable=True)
我根据这个答案创建了一个小部件:
class BinaryFileInputWidget(forms.ClearableFileInput):
def is_initial(self, value):
return bool(value)
def format_value(self, value):
if self.is_initial(value):
return f"{len(value)} bytes"
def value_from_datadict(self, data, files, name):
upload = super().value_from_datadict(data, files, name)
if upload:
return upload.read()
我在
admin.py
中使用它,如下所示:
@admin.register(ImageFile)
class ImageFileAdmin(admin.ModelAdmin):
list_display = ["id"]
formfield_overrides = {
models.BinaryField: {"widget": BinaryFileInputWidget()},
}
然后我在视图中将文件编码为
base64
:
def image_view(request: HttpRequest, id: int):
document_file = ImageFile.objects.filter(id=id).first()
data = ""
if document_file:
data = base64.b64encode(document_file.file).decode(encoding="ascii")
return render(request, "image.html", {"data": data})
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
<body>
<pre>data:image/png;base64,{{ data }}</pre>
<img src="data:image/png;base64,{{ data }}" alt="" />
</body>
</html>
没有错误,数据显示在
pre
元素中,但 img
元素无法加载图像。我尝试使用一些在线转换器来检查数据是否有效,但它们显示错误,指出 base64 字符串无效。
如果我在将文件存储在数据库中之前将文件转换为小部件中的
base64
(return base64.b64encode(upload.read()).decode('ascii')
而不是return upload.read()
),然后在视图中再次转换,它会正确加载。但我不能使用这种方法,因为我可能需要压缩文件,并且显然base64
编码的数据无法正确压缩。
使用我的个人资料图片的示例输出:
<!DOCTYPE html>
<html lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
<body cz-shortcut-listen="true">
<pre>
</pre
>
<img
src=""
alt=""
/>
</body>
</html>
修复模型和视图:
导入base64
from django.shortcuts import render
from .models import ImageFile
def image_view(request, id):
document_file = ImageFile.objects.filter(id=id).first()
data = ""
if document_file and document_file.file: # Ensure `file` is not None
data = base64.b64encode(document_file.file).decode('utf-8')
return render(request, "image.html", {"data": data})
修复 HTML 模板
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Image Viewer</title>
</head>
<body>
{% if data %}
<img src="data:image/png;base64,{{ data }}" alt="Image" />
{% else %}
<p>No image data available.</p>
{% endif %}
</body>
</html>
二进制字段存储:
models.BinaryField 存储原始二进制数据,因此当您调用 upload.read() 在你的小部件中,确保数据没有被修改 (例如,通过部分读取或编码)。
Base64 编码:
Base64 编码必须在渲染图像时进行 在浏览器中,而不是在将数据存储到数据库之前。文件
验证:
确保上传的文件确实是有效的PNG图像。你可以 使用 Pillow 等 Python 库来验证文件类型 上传。
压缩:
首选存储原始二进制数据(而不是预编码的 Base64) 为了效率。 Base64 编码增加存储大小 大约 33%,所以最好只在 检索阶段。