Django在接收到http请求之后,会根据http请求携带的参数以及报文信息创建一个WSGIRequest对象,并且作为视图函数第一个参数传给视图函数。这个参数就是django视图函数的第一个参数,通常写成request。在这个对象上我们可以找到客户端上传上来的所有信息。这个对象的完整路径是django.core.handlers.wsgi.WSGIRequest。
在了解WSGIRequest对象的属性和方法之前,我们先了解一下url的组成,通常来说url的完整组成如下,[]为可选:
protocol 😕/hostname[:port]/path/[;parameters][?query]#fragment
WSGIRequest对象上大部分的属性都是只读的。因为这些属性是从客户端上传上来的,没必要做任何的修改,在django视图中使用时,视图函数的第一个参数参数request就是WSGIRequest对象。以下将对一些常用的属性进行讲解:
path:资源在服务器的完整“路径”,但不包含域名和参数,在url中也是path的内容。比如http://www.baidu.com/xxx/yyy/,那么path就是/xxx/yyy/。
method:代表当前请求的http方法。比如是GET、POST、delete或者是put等方法
GET:一个django.http.request.QueryDict对象。操作起来类似于字典。这个属性中包含了所有以?xxx=xxx的方式上传上来的参数。
POST:也是一个django.http.request.QueryDict对象。这个属性中包含了所有以POST方式上传上来的参数。
FILES:也是一个django.http.request.QueryDict对象。这个属性中包含了所有上传的文件。
COOKIES:一个标准的Python字典,包含所有的cookie,键值对都是字符串类型。
session:一个类似于字典的对象。用来操作服务器的session。
user:user 只有当Django 启用 AuthenticationMiddleware 中间件时才可用。它的值是一个 setting.py 里面AUTH_USER_MODEL 字段所定义的类的对象,表示当前登录的用户。如果用户当前没有登录,user 将设为 django.contrib.auth.models.AnonymousUser 的一个实例。你可以通过 is_authenticated() 区分它们。
META:存储的客户端发送上来的所有header信息,下面是这些常用的header信息:
if request.META.has_key('HTTP_X_FORWARDED_FOR'):
ip = request.META['HTTP_X_FORWARDED_FOR']
else:
ip = request.META['REMOTE_ADDR']
我们平时用的request.GET、request.POST和request.FILES都是QueryDict对象,这个对象继承自dict,因此用法跟dict相差无几。其中用得比较多的是get方法和getlist方法。
users应用,view.py
from django.shortcuts import render
from django.views import View
from django import http
import re
# from users.models import User # 这行提示错误,其实没错,也能用,只是写代码没有提示了。
from meiduo_mall.apps.users.models import User
from django.db import DatabaseError
# Create your views here.
class ResgisterView(View):
""" 用户注册 """
def get(self, request):
""" 用于提供数据--用户注册页面 """
return render(request, 'register.html')
def post(self, request):
""" 实现用户注册业务逻辑:接收参数、校验参数、保存数据、重定向到首页 """
# 接收参数:表单数据
username = request.POST.get('username')
password = request.POST.get('password')
password2 = request.POST.get('password2')
mobile = request.POST.get('mobile')
image_code = request.POST.get('image_code')
sms_code = request.POST.get('sms_code')
allow = request.POST.get('allow')
print(request.POST)
"""# 校验参数,前后端校验需要分开,避免恶意用户避开前端校验直接发起请求,前后端校验逻辑应该相同"""
# 判断参数是否齐全,all函数参数为列表,功能为逐个判定列表元素是否为空,只要有一个为空,返回false,全部非空,返回true
if not all([username, password, password2, mobile, allow]): # 缺少参数
return http.HttpResponseForbidden('必传参数缺失!') # 返回403错误,直接禁止
# 判断用户名是否是5-20个字符
if not re.match(r'^[a-zA-Z0-9_-]{5,20}$', username):
return http.HttpResponseForbidden('用户名为5-20个字符!')
# 判断密码是否是8-20个字符
if not re.match(r'^[a-zA-Z0-9]{8,20}$', password):
return http.HttpResponseForbidden('密码为8-20个字符!')
# 判断两次输入的密码是否一致
if password2 != password:
return http.HttpResponseForbidden('两次输入的密码不一致!')
# 判断手机号码是否合法
if not re.match('^1[3-9]\d{9}$', mobile):
return http.HttpResponseForbidden('手机号不合法!')
# 判断用户是否勾选了协议
if allow != 'on': # checkbox被勾选,发送on给后端
return http.HttpResponseForbidden('请勾选用户协议!')
"""保存注册数据--注册业务核心,调用django的create_user方法。保存数据库,需要用try容错,健壮系统。
保存数据前还需要和数据库中已有的数据核对,用户名、手机号是否重复"""
try:
user = User.objects.create_user(username=username, password=password, mobile=mobile)
except DatabaseError:
return render(request, 'register.html', {'register_errmsg': '注册失败'})
"""注册成功,响应结果,重定向到首页"""
return http.HttpResponse('注册成功,重定向到首页!')
注册失败,register.html的注册按钮后面显示注册失败。
<input type="submit" value="注册">
{% if register_errmsg %}
<span class="error_tip">{{ register_errmsg }}span>
{% endif %}
(meiduo_mall) PS E:\meiduo_project\meiduo_mall> cd meiduo_mall/apps
(meiduo_mall) PS E:\meiduo_project\meiduo_mall\meiduo_mall\apps> python ../../manage.py startapp contents
(meiduo_mall) PS E:\meiduo_project\meiduo_mall\meiduo_mall\apps> cd ..
(meiduo_mall) PS E:\meiduo_project\meiduo_mall\meiduo_mall> cd..
(meiduo_mall) PS E:\meiduo_project\meiduo_mall>


from django.contrib import admin
from django.urls import path, include
from meiduo_mall.apps.users import urls as users_urls
from meiduo_mall.apps.contents import urls as contents_urls
urlpatterns = [
path('admin/', admin.site.urls),
# users
path('', include(users_urls, namespace='users')),
# contents
path('', include(contents_urls, namespace='contents')),
]
from django.urls import path
from . import views
app_name = 'contents'
urlpatterns = [
# 首页广告
path('', views.IndexView.as_view(), name='index'),
]
from django.shortcuts import render
from django.views import View
class IndexView(View):
""" 首页广告 """
def get(self, request):
""" 提供首页广告页面 """
return render(request, 'index.html')
def post(self, request):
pass
"""注册成功,响应结果,重定向到首页"""
# return http.HttpResponse('注册成功,重定向到首页!')
# return redirect('/')
# reverse('contents:index') == '/'
return redirect(reverse('contents:index'))