Python—Django中的视图(views.py)

作者:BerenCamlost

1 前期准备

  1. 启动mysql服务,新建一个数据库
  2. 生成迁移文件
> python manage.py makemigrations
> python manage.py migrate
  1. 启动服务
> python manage.py runserver

2 概述

2.1 作用

  • 视图接受Web请求,并响应Web请求。
  • url只是找到一个视图,真正响应的是视图中的函数。

2.2 本质

视图就是一个Python中的函数

2.3 响应

  1. 网页
    • 重定向
    • 错误视图(404,500,400)详见4.2节
  2. JSON数据

2.4 过程

在这里插入图片描述

3 URL配置

3.1 配置流程

  1. 指定根级url的配置文件
    • 在settings.py文件下的ROOT_URLCONF = 'test1.urls'语句中配置
    • 默认实现了
  2. urlpatterns:这是一个url的实例列表,url对象
    • 正则表达式
    • 视图名称
    • 名称(url反向解析)
from django.contrib import admin
from django.urls import path, include
urlpatterns = [
    path('admin/', admin.site.urls),
    # url(r'^', include('sunck.urls')),  # ^表示什么都不输入,就是主页,加r表示不转义
    # 然后就跳转到sunck.urls.py这个文件中,再在sunck.urls.py文件中配置视图
    path('', include('sunck.urls', namespace='sunck'))
]

from django.conf.urls import url
from django.urls import path
from . import views  # 引入当前目录下的views
app_name = "sunck" 
# 命名app名称,给projests/urls.py中的path('', include('sunck.urls', namespace='sunck'))
# 中的namespace设置可以搜索到的名字
urlpatterns = [
    url(r'^$', views.index),
    path('<int:num>/<int:num2>/', views.detail, name='detail'),
    # def detail(request, num, num2):
    # 		return HttpResponse("detail-%d-%d" % (num, num2))
    path('students/', views.students, name='students'),
    path('grades/<int:num>', views.grades_students, name='gradesStudents'),
]
  1. url配置的注意事项
    • 如果想从url中获取一个值,需要加<>
    • 前边不需要加反斜杠“/”
    • 前边加r表示不转义

3.2 引入其他url配置

  • 在应用中创建urls.py文件,定义本应用的url配置,在工程urls.py中使用include()方法,如上边的代码所示

3.3 url反向解析

  • 【概述】:如果在视图,模板(html)中使用了硬编码链接,在url配置发生改变时,动态生成链接的地址。
  • 【解决思路】:在使用链接时,通过url配置的名称(name),动态生成url地址
  • 【作用】:使用url模板,这个在下一章中会提到

4 视图函数

4.1 定义视图

  1. 【本质】:一个函数
  2. 【视图参数】:
    • 一个HttpRequest实例,浏览器发过来的请求对象
    • 通过正则表达式获取的参数
  3. 位置:一般在views.py中定义

4.2 错误视图

4.2.1 404视图

  • 【概述】:找不到网页(url匹配不成功)时返回
  1. 在templates目录下定义404.html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>404NOT Found</title>
</head>
<body>
    <h1>您请求的页面跑丢了</h1>
    <h2>您请求的页面是:{{ request_path }}</h2>
    <!--request_path是请求的连接-->

</body>
</html>
  1. 配置setings.py文件
DEBUG = False

ALLOWED_HOSTS = ['*']
  • 【注意】这里将DEBUG = False配置后,Django的admin界面的CSS和js文件等都会出现找不到的情况,如果出现类似情况只需要把这个改回来就可以了,但是404页面也就变成了原来的Django样式。

4.2.2 500视图

  • 在视图代码中出现错误(服务器代码错误)

4.2.3 400视图

  • 错误出现在客户的操作

5 HttpRequest

5.1 概述

  1. 服务器接收Http请求后,会根据报文创建HTTP Request对象
  2. 视图的第一个参数是HTTP Request对象
  3. Django创建的,视图调用时传递给视图

5.2 属性

path:		请求的完整路径(不包括域名和端口)
method:		表示请求的方式,常用的有GET,POST
encoding:	表示浏览器提交的数据的编码方式,一般为utf-8
GET:		类似于字典的对象,包含了get请求的所有参数
POST:		类似于字典的对象,包含了post请求的所有参数
FILES:		类似字典的对象,包含了所有上传的文件
COOKIES:	字典,包含所有的cookie
session:	类似字典的对象,表示当前会话
  • 利用网页显示这些信息:
# urls.py
path('request/', views.requests, name='request'),

# views.py
def requests(request):
    request_list = {
        'request.path': request.path,
        'request.method': request.method,
        'request.encoding': request.encoding,
        'request.GET': request.GET,
        'request.POST': request.POST,
        'request.FILES': request.FILES,
        'request.COOKIES': request.COOKIES,
        'request.session': request.session,
        'request.is_ajax()': request.is_ajax(),
    }
    return render(request, 'sunck/request.html', {'request_list': request_list})
<!--sunck/request.html-->
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>HTTPrequest属性和方法测试</title>
</head>
<body>
    <h1>HTTPrequest属性和方法测试</h1>
    <ul>
        {% for key, value in request_list.items %}
            <li>{{ key }}:{{ value }}</li>
        {% endfor %}
    </ul>

</body>
</html>

5.3 方法

is_ajax():	如果是通过XMLHttpRequest发起的,返回 True

5.4 QueryDict对象

  • request对象中的GET,POST都属于QueryDict对象

方法

  1. get():
    根据键获取值,只能获取一个值
    www.sunck.wang/abc?a=1&b=2&c=3
  2. getlist()
    将键的值以列表的形式返回
    可以获取多个值
    www.sunck.wang/abc?a=1&a=2&c=3

5.5 GET属性

5.5.1 作用

获取浏览器传递给服务器的数据

5.5.2 get()

每个键只能对应一个数据
http://127.0.0.1:8000/sunck/GET/?a=1&b=2&c=3

# 获取get传递的数据
def gets(request):
    get_list = {
        'a': request.GET.get('a'),
        'b': request.GET['b'],  # 也可以用这种方式
        'c': request.GET.get('c'),
    }
    return render(request, 'sunck/GET.html', {'get_list': get_list})

HTML文件:“sunck/GET.html”

{#sunck/request.html#}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>GET属性和方法测试</title>
</head>
<body>
    <h1>GET属性和方法测试</h1>
    <ul>
        {% for key, value in get_list.items %}
            <li>{{ key }}:{{ value }}</li>
        {% endfor %}
    </ul>
</body>
</html>

5.5.3 getlist()

每个键可以对应多个数据,对应一个数据列表
http://127.0.0.1:8000/sunck/GET/?a=1&a=2&c=3

ef gets(request):
    get_list = {
        'list': request.GET.getlist('a'),
        'c': request.GET.get('c'),
    }
    return render(request, 'sunck/GET.html', {'get_list': get_list})

对应的HTML文件和上边的一样

5.6 POST

5.6.1 属性

利用表单模拟POST请求。
下面简单介绍使用一个注册的页面,传递数据给另外一个界面。

5.6.2 HTML文件

1 sunck/POST/show_register.html

这个页面是用户看到的注册页面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>注册</title>
</head>
<body>
<form action="../register/" method="post">
<!--这里的action需要和后面的urls路径相对应-->
    姓名:<input type="text" name="name" value=""/>
    <hr>
    性别:<input type="radio" name="gender" value="1"><input type="radio" name="gender" value="0"><hr>
    年龄:<input type="text" name="age" value=""/>
    <hr>
    爱好:<input type="checkbox" name="hobby" value="power"/>权利<input type="checkbox" name="hobby" value="money">金钱<input type="checkbox" name="hobby" value="beauty">美女<input type="checkbox" name="hobby" value="Tesla">Tesla
    <hr>
    <input type="submit" value="注册">
</form>
</body>
</html>

界面效果如下图所示:
注册界面

2 sunck/POST/register.html

这个页面是为了将用户注册的数据显示出来,其原理和register、get测试时的原理相同。

{#sunck/request.html#}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>POST属性和方法测试</title>
</head>
<body>
    <h1>POST属性和方法测试</h1>
    <ul>
        {% for key, value in register_list.items %}
            <li>{{ key }}:{{ value }}</li>
        {% endfor %}
    </ul>
</body>
</html>

5.6.3 views.py视图文件

# POST测试
def show_register(request):
    return render(request, 'sunck/POST/show_register.html', )


def register(request):
    register_list = {
        'name': request.POST.get('name'),  # 这里的get和getlist可能没有代码提示
        'gender': request.POST.get('gender'),
        'age': request.POST.get('age'),
        'hobby': request.POST.getlist('hobby'),
    }
    return render(request, 'sunck/POST/register.html', {'register_list': register_list}, )

5.6.4 urls.py路径

    path('POST/showRegister/', views.show_register, ),
    path('POST/register/', views.register, ),

5.6.5 修改settings.py文件

注释掉那一行
因为Django有自动预防跨站攻击

MIDDLEWARE = [
    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    # 'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

6 HTTP Response对象

6.1 概述

  • 【作用】:给浏览器返回数据
  • 【对比】:HttpRequest对象是由Django创建的,HttpResponse对象是由程序员创建

6.2 返回数据用法

6.2.1 不用模板,直接返回数据

def index(request):
	return HttpResponse('sunck is a good man')

6.2.2 调用模板,使用render方法

  1. 原型
def render(request, template_name, context=None, 
content_type=None, status=None, using=None):
# request:请求体 对象
# template_name:模板路径
# context:传递给需要渲染在模板上的数据
  1. 作用:
    结合数据和模板,返回一个完整的HTML页面
  2. 示例:
def grades_students(request, num):
    # 获得对应的班级对象
    grade = Grades.graObj2.get(pk=num)
    # 获得班级下的所有学生列表
    students_list = Students.stuObj2.filter(sgrade_id__exact=num)
    return render(request, 'sunck/students.html', {
                    "students": students_list,
                    "grade": grade.gname,
                  })

6.3 属性

  1. content:表示返回内容
  2. charset:返回数据的编码格式
  3. status_code:响应状态码
    200,304,404
  4. content-type:指定输出的MIME类型
# response
def response(request):
    response1 = HttpResponse()
    response_list = {
        'response.content': response1.content,
        'response.charset': response1.charset,
        'response.status_code': response1.status_code,
    }
    return render(request, 'sunck/response.html', {'response_list': response_list})

6.4 方法

  1. init:使用页面内容实例化HttpResponse对象
  2. write(content):以文件的形式写入
  3. flush():以文件的形式输出缓冲区
  4. set_cookie(key, value, maxAge=None, exprise=None)
    设置一条cookie
  5. delete_cookie(key):
    • 删除cookie
    • 如果删除一个不存在的cookie,就当什么都没发生

6.5 子类:HttpResponseRedirect

6.5.1 功能

重定向,服务器端的跳转

6.5.2 示例

# 重定向
from django.http import HttpResponseRedirect


def redirect1(request):
    # 可以使用如下两种方式
    # return HttpResponseRedirect('/sunck/redirect2/')
    return HttpResponseRedirect('../redirect2/')

def redirect2(request):
    return HttpResponse('我是重定向后的视图')

6.5.3 简写

使用redirect(to)这个方法,在django.shortcuts中包含

# 重定向
from django.shortcuts import redirect

def redirect1(request):
    # 简写
    return redirect('/sunck/redirect2')

def redirect2(request):
    return HttpResponse('我是重定向后的视图')

to推荐使用反向解析

6.6 子类:JsonResponse

  1. 返回Json数据,一般用于异步请求
  2. __init__(self.data)
  3. data:一个字典对象
  4. 注意:Content-type类型为application/json

7 状态保持

7.1 概述

  • http协议是无状态的,每次请求都是一次新的请求,它不记得之前的请求。
  • 客户端与服务器的一次通信就是一次会话
  • 实现状态的保持,在客户端或服务端存储有关会话的数据
  • 存储的方式
    • cookie:所有数据存储在客户端,不要存储敏感的数据
    • 【推荐】session:所有的数据存储在服务端,在客户端用cookie存储session_id
  • 状态保持的目的:
    在一段时间内跟踪请求者的状态,可以实现跨页面访问当前的请求者的数据
  • 注意:不同的请求者之间不会共享这个数据,与请求者一一对应

7.2 启用session

在settings.py中找到
session1
session2
默认情况下,Django是启用session的。

7.3 使用session

  1. 启用session后,每个httpRequest对象都有一个session属性,就是一个类似字典的对象
  2. get(key, default=None) 根据键获取session值
  3. clear()
    清空所有会话
  4. flush()
    删除当前会话并删除会话的cookie
# session
from django.contrib.auth import logout
def welcome(request):
    username = request.POST.get('username')
    # 储存session
    request.session['username'] = username
    # 取session
    username = request.session.get('username', '游客')
    # 清除session
    # logout(request)
    # request.session.clear()
    # request.session.flush()
    return render(request, 'sunck/session/welcome.html', {'username': username})


def log(request):
    return render(request, 'sunck/session/log.html', )

对应的HTML文件为

  • welcome.html:
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>我的</title>
</head>
<body>
    <h1>欢迎:{{ username }}</h1>
    <a href="/sunck/session/log/">登录</a>
</body>
</html>
  • log.html:
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>登录</title>
</head>
<body>
    <form action="/sunck/session/welcome/" method="post">
        <input type="text" name="username"/>
        <input type="submit" value="登录"/>
    </form>
</body>
</html>

7.4 设置过期时间

  • 使用set_expiry(value)设置过期时间
    request.session.set_expiry(10) 设置为10秒后过期
  • 如果不设置,2个星期后过期
  • 时间对象
  • value设置为0代表关闭浏览器时过期
  • value设置为None代表设置永不过期,不推荐

下一章:Python—Django中的模板(template)

猜你喜欢

转载自blog.csdn.net/weixin_43830248/article/details/86626031