Django 实现将图片转为Base64,然后使用json传输
作者:loovelj 发布时间:2023-06-14 06:42:28
最近使用Django来进行图像的传输,由于要求需要使用Json格式进行请求,所以我们尝试了二进制编码放在json里,发现bytes格式不能打入json,于是转为了base64
将图片转为json
图片转为json有2中方法,一个是二进制后再转,一个是转为矩阵以后再转。
import base64
import cv2
#通过opencv转base64
img_im= cv2.imread("D://32.png")
aa=base64.b64encode(cv2.imencode('.jpg',img_im)[1]).decode()
print(len(aa)) #17292
#通过bytes再转base64
bb=base64.b64encode(open("D://32.png", 'rb').read())
print(len(bb)) #43848
最后采用了Opencv的方式,主要发现opencv的base64编码比第二种短一半,所以用了第一种。
客户端请求发送数据格式
客户端请求服务器的base64,目前我这里发现2中,通过Postman测试。
第一种 ‘form/data':
#客户端:
image1= cv2.imread("D://32.png")
aa=base64.b64encode(cv2.imencode('.jpg',image1)[1]).decode()
r = requests.post(url, data={"image": image1})
print(r.content.decode("utf-8"))
#服务端:
def image_base64(request):
result = request.POST.get("image")
img_byte = base64.b64decode(result)
img_np_arr = np.fromstring(img_byte, np.uint8)
image = cv2.imdecode(img_np_arr, cv2.IMREAD_COLOR)
#image 已经转为矩阵了
第二种 ‘application/json':
#客户端:
image1= cv2.imread("D://32.png")
aa=base64.b64encode(cv2.imencode('.jpg',image1)[1]).decode()
r = requests.post(url, json={"image": image1})
print(r.content.decode("utf-8"))
#服务端:
def local_ocr_base64(request):
# result = request.POST.get("image")
data = request.body
data_json=simplejson.loads(data) #data是str格式的,需要转为json
result=data_json["image"]
img_byte = base64.b64decode(result)
img_np_arr = np.fromstring(img_byte, np.uint8)
image = cv2.imdecode(img_np_arr, cv2.IMREAD_COLOR)
补充知识:Django将多个图片保存成一个URL串返回给前端
说明
Django有ImageField字段,是封装好的,使用很方便,但是一个ImageField字段只能保存一张图片的URL,我现在是想将多张图片保存在一个字段里,然后URL和URL之间用一个自定义的分隔符连接起来,这样不用为了图片再设计一个字表。
自定义上传图片
设置settings.py文件,添加如下语句
MEDIA_URL = '/media/'
MEDIA_ROOT = os.path.join(BASE_DIR, 'media/')
IMAGE_ROOT = os.path.join(MEDIA_ROOT, 'images/')
WEB_HOST_MEDIA_URL = os.path.join('http://127.0.0.1:8000', MEDIA_URL[1:], 'images/')
MEDIA_ROOT代表的是用户上传后的文件一般保存的地方,一般在放在项目目录下,例如BASE_DIR就是取得项目绝对地址。而MEDIA_URL是指URL访问时的URL。例如,
BASE_DIR: /Users/incisor/VSCodeProjects/python/Notes
那么
MEDIA_ROOT:/Users/incisor/VSCodeProjects/python/Notes/media
假设media目录下有一张图片2019.jpg,那么
MEDIA_URL: http://127.0.0.1:8000/media/
然后再通过http://127.0.0.1:8000/media/2019.jpg这个URL是可以直接访问这个图片的。
IMAGE_ROOT是我自定义的一个字段,因为我想以后可能会上传视频,或者其他一些文件,想区分开,所以我在media目录下再建了一个目录images,那么
IMAGE_ROOT: /Users/incisor/VSCodeProjects/python/Notes/media/images
WEB_HOST_MEDIA_URL是拼接URL时使用,因为我接下来要自己拼接多个图的URL,MEDIA_URL[1:]的原因时如果不去掉第一个/,那WEB_HOST_MEDIA_URL会是/media/images/,这显然不是我们想要的,所以需要把第一个/去掉。
views.py文件
前端传过来不定数量的图片,可能0张,最多9张,每个图片都做base64编码再传过来,POST请求里有个参数imgs,是一个数组,保存base64编码。
base64编码如下图所示,这只是一部分:
前面的data:image/png;base64, 除了后缀会不同,其他的是固定的,所以我会先从这里取得后缀png,再做编码转换。
import base64
import os
import re
from datetime import date, datetime
from rest_framework import status
from rest_framework.decorators import api_view
from rest_framework.response import Response
from Notes.settings import IMAGE_ROOT, WEB_HOST_MEDIA_URL
@api_view(['POST',])
def images(request):
urls = ''
dir_name = date.today().__str__().replace('-', '_', 2) # 2019_06_21
dirs = os.path.join(IMAGE_ROOT, dir_name) # 将日期作为目录名
if not os.path.isdir(dirs):
os.makedirs(dirs) # 判断目录是否存在,不存在则创建
for img in request.data['imgs']:
strs = img.split(',')
suffix = re.findall(r'/(\w+?);', strs[0])[0] # 取得文件后缀
# 拼接服务器上的文件名
# datetime.now()取得当前时间,精确到了微秒,一般来说是唯一的了,因为目录是日期,所以文件名就去掉日期,最后会是一串数字
img_name = re.sub(r':|\.', '', datetime.now().__str__().split(' ')[1]) + '.' + suffix
img_path = os.path.join(dirs, img_name)
with open(img_path, 'wb') as out:
out.write(base64.b64decode(strs[1])) # base64解码,再写入文件
out.flush()
urls += os.path.join(WEB_HOST_MEDIA_URL, dir_name, img_name) + '[/--sp--/]' # 拼接URL,URL与URL之间用[/--sp--/]隔开
result = {}
result['status'] = status.HTTP_200_OK
result['message'] = '图片上传成功'
result['urls'] = urls[:len(urls) - len('[/--sp--/]')] # 去掉末尾的[/--sp--/]
return Response(data=result)
3、urls.py文件
from django.contrib import admin
from django.urls import path
from django.conf.urls.static import static # 需要添加这句,包含静态资源之类的
from note import views
from . import settings
urlpatterns = [
path('notes/images/', views.images),
path('admin/', admin.site.urls),
]
urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT) # 这句不设置的话,不能通过URL直接访问到图片
4、启动服务,python3 manage.py runserver
5、POST请求,然后返回对应的JSON数据
{
"status": 200,
"message": "图片上传成功",
"urls": "http://127.0.0.1:8000/media/images/2019_06_21/073249204253.png[/--sp--/]http://127.0.0.1:8000/media/images/2019_06_21/073249854323.png"
}
这样在数据库方面,就可以直接用一个varchar类型来存储多张图了,前端收到URL串,再按照定好的分隔符[/–sp–/]切开后,就可以显示了。
来源:https://blog.csdn.net/loovelj/article/details/92833070


猜你喜欢
- 核心技术:Python3.7GUI技术:Tkinter (Python已经内置)好多文章写Python GUI之tkinter窗口视窗教程大
- 在第一章,我们讲过position,当它取值absolute的时候,即绝对定位。既然是定位,就要有参照物。一般来讲,这个定位的参照物是bod
- 在Class内部,可以有属性和方法,而外部代码可以通过直接调用实例变量的方法来操作数据,这样,就隐藏了内部的复杂逻辑。但是,从前面Stude
- 1. 指针区别于C/C++中的指针,Go语言中的指针不能进行偏移和运算,是安全指针。要搞明白Go语言中的指针需要先知道3个概念:指针地址、指
- 暂时性死区只要块级作用域存在let命令,它所声明的变量就“绑定”这个区域,不再受外部的影响。这么说可能有些抽象,举个例子:var
- Go提供几种方法检查变量的类型,在字符串格式化标识%T, 反射方式:reflect.TypeOf, reflect.ValueOf.Kind
- 这是Pyhon系列文章的第二篇,本文主要介绍如何运行Python程序以及安装PyCharm开发工具。干货满满,建议收藏,需要用到时常看看。
- 作为一名前端,我们通常要做的就是让页面在各系统A-Grade浏览器,甚至网站浏览份额0.1%以上的浏览器上良好显示。当然,还有性能问题。不过
- Python 是一个美丽的语言,可以激发用户对它的爱。所以如果你试图加入程序员行列,或者你有点厌倦C++,Perl,Java 和其他语言,我
- 淘宝招聘的一个css题目:css样式控制div水平垂直居中方法;<!DOCTYPE html PUBLIC "-//W3C/
- 讲解1、库:os,shutil.copy2、代码效果:对指定文件夹内文件等量分配到新的文件夹3、代码原理:用os.listdir()遍历文件
- 环境python3.6合并前第一个文件和第二个文件代码file1 = open("my.txt","r&quo
- 本文转自:https://blog.csdn.net/qq_42730750/article/details/108415551前言 各大
- 1、panic当我们执行panic的时候会结束下面的流程:package mainimport "fmt"func ma
- 前言本文的主要内容是使用 cpu 版本的 tensorflor-2.1 完成对 Auto MPG 数据集的回归预测任务。获取 Auto MP
- 引言自ChatGPT出现,各种基于它的软件扩展纷至沓来,目前Word支持ChatGPT的add-in有两款,可以通过:插入->获取加载
- 有的时候,操作大文件,或者取数,要很久,我们给脚本首尾添加一段代码就知道,这段代码整体的大致运行时间了。import timestart =
- 第一种情况:有RAID,还需要做数据库备份吗?回答:需要。有了RAID,万一部份磁盘损坏,可以修复数据库,有的情况下数据库甚至可以继续使用。
- 利用FSO取得BMP,JPG,PNG,GIF文件信息:大小,宽、高尺寸等 ’:::::::::::::::::::::::::::
- 模型事件Laravel 模型事件允许你监听模型生命周期内的事件, 并且通过这个事件去做一些模型通用性的东西, 例如检查用户修改了那个字段,