请求与响应
简单流程图
我们先来了解一个请求与响应的大概流程
视图函数接受到的request到底是个什么对象呢?
服务器接收到http协议的请求后,会根据报文创建HttpRequest对象视图函数的第一个参数是HttpRequest对象在django.http模块中定义了HttpRequest对象的API,下面我们来列出这个对象的方法。
request属性 | 说明 |
---|---|
path | 一个字符串,表示请求页面的完整路径,不包括域名 |
method | 一个字符串,表示请求使用的HTTP方法,常用值包括:'GET'、'POST' |
*encoding | 一个字符串,表示提交的数据的编码方式, |
GTE | 一个类似于字典的对象,包含get请求方式的所有参数 |
POST | 一个类似于字典的对象,包含post请求方式的所有参数 |
FILES | 一个类似于字典的对象,包含所有的上传文件 |
COOKIES | 一个标准的python字典,包含所有的cookie,键和值都为字符串 |
sessin | 一个即可读有可写的类似字典的对象,表示当前的会话,只有当Django启动会话的支持时才可用,详细内容见"状态保存" |
方法:is_ajax() | 如果请求是通过XMLHttpRequest发起的,则返回True |
*encoding:如果为None则表示使用浏览器的默认设置,一般为utf-8,这个属性是可写的,可以通过修改它来修改访问表单数据使用的编码,接下来对属性的任何访问将使用新的encoding值
POST登录提交的html例子
html中的form表单解析
<form action="提交地址,为空为当前页" method="默认为POST请求"> # 如果是post请求,会有安全的问题,需要加上csrf的方法,才可以使用post请求。
{% csrf_token %} <p>用户名:<input type="text" name="必须要给的属性,用于后台获取参数得到的key"></p>
<p>密码:<input type="password" name="必须要给的属性,用于后台获取参数得到的key"></p>
<p>用户名:<input type="submit" value="登录"></p>
</form>
views中的简单逻辑判断写法:
因为登录页面有GET请求,也有POST的请求,所以需要再视图的时候需要进行判断,做出反应。
def login(request): if request.method =='POST' # userame和password 这个key是前端网页传过来的类似的字典方式。
username = request.POST.get('username','')
password = request.POST.get('password','') # 如果账号和密码是对的,则跳转到首页
if username == 'aaa' and password == '':
return redirect(reverse('teacher:index'))
else:
此处添加账户密码错误的返回结果 # 如果是GET方法则把login页面返回过去
return render(request, 'teacher/login.html')
get以及post方式访问如何的获取多个参数
当网页是get访问的时候,且是多个参数的时候,取值是什么样的呢。
访问的网页url例子:
http://127.0.0.1:8000/teacher/login/?hobby=篮球&hobby=足球&hobby=羽毛球
正常的取值方法,只能获取到一个值
#一键多值的情况下,只能拿到一个值
>>> request.GET.get('hobby')
'羽毛球'
getlist:拿多个值的时候,需要getlist
>>> request.GET.getlist('hobby')
['篮球','足球','羽毛球']
POST方法相同,只需要修改'GET'为'POST'即可。
文件上传
需要设置上传文件存放的路径,setting中设置,同时需要新建文件夹。
#测试用的,名字可以随意取,暂时路径为根目录下的文件夹。
UPLOAD_ROOT = os.path.join(BASE_DIR,'upload')
html中form的基础写法
<form action="" method="post" enctype="multipart/form-data">
{% scrf_token %}
<p>请选择你要上传的文件:<input type="file" name="file" multiple></p>
<p><input type="submit" value="上传"></p>
</form> enctype="multipart/form-data":只有form表带拥有这个属性的时候,才有上传文件的功能
multiple:选择文件需要带有才可以上传多个文件
配置view: 判断file,如果有文件,则创建文件夹(文件夹名为当天日期)
from datetime import datetime
from crm.setting import UPLOAD_ROOT
import os def upload(request):
if request.method == 'POST':
files = request.FILES.getlist('file', None)
#如果有多个文件,则循环取出,
for file in files:
#文件名
day_dir = datetime.noe.strftime('%Y%m%d')
#绝对路径拼接
pre_dir = os.path.join(UPLOAD_ROOT,day_dir) #判断文件夹是否存在,需要注意当前用户是否有权限
if not os.path.exists(pre_dir):
os.mkdir(pre_dir) #如果不存在则创建文件夹 # 拿到用户上传过来的文件名,和路径拼接
filename = os.path.join(pre_dir, file.name) # 写数据(二进制)
with open(filename, 'wb') as f:
#chunks是一个以文件流的形式取值。如果文件较大,以read的方式取值则会耗费资源。
for line in file.chunks():
f.write(line)
return HttpResponse('上传成功') return render(request, 'teacher/upload.html')
HTTPResponse对象:
from django.http import HttpResponse
属性 | 作用 |
---|---|
content | 表示返回的内容,字节类型 |
charset | 表示response采用的编码字符集,字符串类型 |
status_code | 响应的HTTP响应状态码 |
content-type | 指定输出的MIME类型,请求头 |
方法 | 作用 |
init | 使用页内容实例化HttpResponse对象 |
write(content) | 以文件的方式写,可以分次写 |
flush() | 以文件的方式输出缓存区 |
set_cookie(key, value='', max_age=None, expires=None) | 设置Cookie |
key、value | 都是字符串类型 |
max_age | 是一个整数,表示在指定数秒后过期 |
expires | 是一个datetime或timedelta对象,会话将在这个指定的日期/时间过期。 |
max_age与expires二选一 | max_age与expires二选一,如果不指定过期时间,则关闭浏览器就失效 |
deleta_cookie(key) | 删除指定的key的cookie,如果key不存在则数秒也不发送 |
*expires:注意datetime和timedelta值只有在使用PickleSerializer时才可序列化。
JsonResponse 返回json数据,只能接受字典类型
如何返回一个json数据,这里先演示一个查询数据的接口
def students_api(request):
sex = request.GET.get('sex','')
sex = int(sex)
res = Students.objects.values('name','age','sex','phone').filter(sex=sex)
res = dect('data':res)
return JsonResponse(res)