接上节:入门【一】
再创建一个orders子应用,python manager.py startapp orders,orders目录中新建一个urls.py文件。结构如图:
通过上节课,我们知道在views.py文件中编写函数时,有一个默认入参request,通过它我们可以获取前端传进来的数据。
获取request请求参数
一、获取url后的查询参数
在orders/views.py中新增index函数,直接打印request的内容及其它输出方法。
浏览器中访问如下地址:
http://127.0.0.1:1992/orders/index/?name=testyan&class=yuwen&class=englist
# orders/views.py
from django.http import HttpResponse, JsonResponse
def index(request):
print(request) # <WSGIRequest: GET '/orders/index/?name=testyan&class=yuwen&class=englist'>
print(request.GET) # <QueryDict: {'name': ['testyan'], 'class': ['yuwen', 'englist']}>
print(request.GET.get('name')) # testyan
print(request.GET.getlist('class')) # ['yuwen', 'englist']
print(request.GET.get('sex', 'F')) # F 当无字段值时,第二个字段为默认值
request.GET仅获取url后拼接的数据,在浏览器中通过地址栏访问是GET方法,但其实无论是何种请求方式都能通过request.GET获取,比如你可以在postman中选择POST发送请求,也能获取到请求url后拼接的查询参数。
二、获取请求数据
print(request.POST)
print(request.POST.get('fruit'))
print(request.POST.getlist('num'))
request.POST只能接收post方法的请求体,不能接受put或patch方法,而且只能针对form-data 和x-www-form-urlencoded格式才能获取到。如果是json等其它格式,可以通过request.body获取。
如下代码是在postman中传入json格式的入参获取的结果:
print(request.body) # b'{"fruit":"apple",\r\n"num":10\r\n}'
print(json.loads(request.body)) # {'fruit': 'apple', 'num': 10}
接受post、put或patch方法的请求体都可用request.body,如果是表单请求,数据格式比较冗余,建议使用json格式的请求数据。
三、获取请求头信息
在postman中请求时,可以自定义请求头字段,通过以下方式获取。
# 获取原生请求头信息
print(request.META)
# 获取处理后的请求头信息
print(request.headers)
# 获取自定义头信息
print(request.headers.get('define'))
四、获取请求体中的文件
from django.views.decorators.http import require_http_methods
# Create your views here.
@require_http_methods(['POST'])
def index(request):
# 获取请求中的文件,只有POST请求才可以获取
file = request.FILES
print(file) # <MultiValueDict: {'fruit': [<InMemoryUploadedFile: windows秘钥.txt (text/plain)>]}>
print(file.get('fruit').read()) # 读取文件内容
return HttpResponse('<h1>orders的index页面</h1>', content_type='text/html')
注意,只有POST请求才能获取文件数据。
获取response响应参数
以上代码示例都是返回一个普通的文字或样式,除此外,还可以返回json数据、图片、压缩包等。
一、返回json数据
from django.http import HttpResponse, JsonResponse
def index2(request):
# 可以定义content_type、status和响应头
# return HttpResponse(content='<h1>orders2</h1>', content_type='text/html', status=201, headers={'token': 80})
data = {'name': 'siwan', 'age': 10}
# return HttpResponse(content=json.dumps(data), content_type='text/json')
# return JsonResponse(data) # 同上一行的效果
data2 = [{'name': 'siwan', 'age': 10}, {'name': 'bawan', 'age': 20}]
# 因为data2其实是列表,当用JsonResponse返回时,需要关闭安全检测。
return JsonResponse(data2, safe=False)
二、返回图片或压缩文件
def index3(request):
# 返回图片、压缩文件等
with open('./image-1.png', 'rb') as f:
img = f.read()
return HttpResponse(img, content_type='image/png')
# with open('./zip.zip', 'rb') as f:
# zip = f.read()
# return HttpResponse(zip, content_type='application/zip')
重定向功能实现
一、外部重定向
def index4(request):
# 外部重定向,302临时重定向
# res = HttpResponse(status=302)
# res['location'] = 'https://www.qq.com'
# return res
# 相当于
# return HttpResponseRedirect('https://www.qq.com')
当访问 http://127.0.0.1:1992/orders/index4/时,就自动跳转到qq网站。
二、内部重定向
# 内部重定向, 基于redirect直接指定内部跳转路由
def index4(request):
from django.shortcuts import redirect
return redirect('/orders/index3/')
以上设置,会讲index4的访问重定向到index3。但是当正则路由时,这样写就不太方便,需要给路由设置个别名。正则路由指的是路由地址不是固定的,而是需要用正则来匹配。
# 首先在orders/urls.py中给index3设置路由别名,并给orders设置app_name
from . import views
from django.urls import path
app_name = 'orders_alias'
urlpatterns = [
path('index/', views.index),
path('index2/', views.index2),
path('index3/', views.index3, name='alias3'), # 路由别名
path('index4/', views.index4)
]
# 然后在跟urls.py中给orders设置命名空间,要和app_name一致
from django.contrib import admin
from django.urls import path, include
urlpatterns = [
path('admin/', admin.site.urls),
path('app01/', include('app01.urls')),
path('orders/', include('orders.urls', namespace='orders_alias'))
]
# 然后orders/views.py中
def index4(request):
# 内部重定向,基于redirect+reverse对路由别名反向解析进行跳转
from django.shortcuts import redirect, reverse
url = reverse('orders_alias:alias3')
print(url)
return redirect(url)
这样,当访问http://127.0.0.1:1992/orders/index4/ 时就会重定向到 http://127.0.0.1:1992/orders/index3/