django文档7-8

# 七、Ajax请求 客户端(浏览器)向服务端发起请求的形式: 1. 地址栏:GET 2. 超链接标签:GET 3. form表单:GET或POST 4. Ajax(重要):GET或POST或PUT或DELETE AJAX(Asynchronous Javascript And XML)翻译成中文就是“异步Javascript和XML”。即使用Javascript语言与服务器进行异步交互,传输的数据为XML(当然,传输的数据不只是XML,现在更多使用json数据)。 AJAX的特点和优点: - 异步 - 局部刷新 应用: ![image-20210812104733820](assets/image-20210812104733820.png) ## 7.2、json数据 ```python ''' Supports the following objects and types by default: +-------------------+---------------+ | Python | JSON | +===================+===============+ | dict | object | +-------------------+---------------+ | list, tuple | array | +-------------------+---------------+ | str | string | +-------------------+---------------+ | int, float | number | +-------------------+---------------+ | True | true | +-------------------+---------------+ | False | false | +-------------------+---------------+ | None | null | +-------------------+---------------+ ''' ``` #### (1)python的序列化和反序列化方法 ```python import json dic = {"name": "yuan"} dic_json = json.dumps(dic) dic = json.dumps(dic_json) ``` #### (2)Django支持的序列化方法 关于Django中的序列化主要应用在将数据库中检索的数据返回给客户端用户,特别的Ajax请求一般返回的为Json格式。 ```python # 序列化响应类 from django.http import JsonResponse JsonResponse({}) # 序列化queryset from django.core import serializers ret = models.Book.objects.all() data = serializers.serialize("json", ret) ``` ## 7.3、Ajax请求案例 ![image-20210812113615978](assets/image-20210812113615978.png) #### (1) 视图 ```python # Create your views here. def reg(request): return render(request, "reg.html") def reg_user(request): data = {"msg": "", "state": "success"} user = request.POST.get("user") if user == "yuan": data["state"] = "error" data["msg"] = "该用户已存在!" return JsonResponse(data) ``` #### (2) 模板:reg.html ```html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p>用户名:<input type="text"><span class="error" style="color: red"></span></p> <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.0/jquery.js"></script> <script> $(":text").blur(function () { // 发送Ajax请求 $.ajax({ url: "http://127.0.0.1:8008/reg_user/", type: "post", data: { user: $(":text").val(), }, success: function (response) { console.log(response); $(".error").html(response.msg); } }) }) </script> </body> </html> ``` #### (3) 流程图 ![image-20210812114853512](assets/image-20210812114853512.png) 练习: 1. 计算器 2. ajxa通过orm获取书籍并展示 ## 7.4、同源策略 ### 7.4.1、同源策略和跨域 现在我们将reg.html单独放在客户端,用浏览器打开,再触发事件,会发现报错: ![image-20210812115456734](assets/image-20210812115456734.png) 这是因为浏览器的同源策略导致的。 同源策略(Same origin policy)是一种约定,它是浏览器最核心也最基本的安全功能,如果缺少了同源策略,则浏览器的正常功能可能都会受到影响。可以说Web是构建在同源策略基础之上的,浏览器只是针对同源策略的一种实现。 > 同源策略,它是由Netscape提出的一个著名的安全策略。现在所有支持JavaScript 的浏览器都会使用这个策略。所谓同源是指,域名,协议,端口相同。当一个浏览器的两个tab页中分别打开来 百度和谷歌的页面当浏览器的百度tab页执行一个脚本的时候会检查这个脚本是属于哪个页面的,即检查是否同源,只有和百度同源的脚本才会被执行。如果非同源,那么在请求数据时,浏览器会在控制台中报一个异常,提示拒绝访问。 那么如何解决这种跨域问题呢,我们主要由三个思路: > 1. jsonp > 2. cors > 3. 前端代理 这里主要给大家介绍第二个:cors CORS需要浏览器和服务器同时支持。目前,所有浏览器都支持该功能,IE浏览器不能低于IE10。 整个CORS通信过程,都是浏览器自动完成,不需要用户参与。对于开发者来说,CORS通信与同源的AJAX通信没有差别,代码完全一样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感觉。 因此,实现CORS通信的关键是服务器。只要服务器实现了CORS接口,就可以跨源通信。 所以,服务器方面只要添加一个响应头,同意跨域请求,浏览器也就不再拦截: ```python response = JsonResponse(data) response["Access-Control-Allow-Origin"] = "*" ``` ![image-20210812121643296](assets/image-20210812121643296.png) ### 7.4.2、cors cors有两种请求:简单请求和非简单请求 只要同时满足以下两大条件,就属于简单的请求 ```text (1) 请求方法是以下三种方法之一: HEAD GET POST (2)HTTP的头信息不超出以下几种字段: Accept Accept-Language Content-Language Last-Event-ID Content-Type:只限于三个值application/x-www-form-urlencoded、multipart/form-data、text/plain ``` 凡是不同时满足上面两个条件,就属于非简单请求。浏览器对这两种请求的处理,是不一样的。 简单请求:一次请求 非简单请求:两次请求,在发送数据之前会先发一次请求用于做“预检”,只有“预检”通过后才再发送一次请求用于数据传输。 ``` - 请求方式:OPTIONS - “预检”其实做检查,检查如果通过则允许传输数据,检查不通过则不再发送真正想要发送的消息 - 如何“预检” => 如果复杂请求是PUT等请求,则服务端需要设置允许某请求,否则“预检”不通过 Access-Control-Request-Method => 如果复杂请求设置了请求头,则服务端需要设置允许某请求头,否则“预检”不通过 Access-Control-Request-Headers ``` 支持跨域,简单请求: ```text 服务器设置响应头:Access-Control-Allow-Origin = '域名' 或 '*' ``` 支持跨域复杂请求: ``` 由于复杂请求时,首先会发送“预检”请求,如果“预检”成功,则发送真实数据。 “预检”请求时,允许请求方式则需服务器设置响应头:Access-Control-Request-Method “预检”请求时,允许请求头则需服务器设置响应头:Access-Control-Request-Headers ``` cors在Django中的实现: 在返回结果中加入允许信息(简单请求): ```python def test(request): import json obj=HttpResponse(json.dumps({'name':'yuan'})) # obj['Access-Control-Allow-Origin']='*' obj['Access-Control-Allow-Origin']='http://127.0.0.1:8004' return obj ``` 放到中间件中处理复杂和简单请求: ```python from django.utils.deprecation import MiddlewareMixin class MyCorsMiddle(MiddlewareMixin): def process_response(self, request, response): # 简单请求: # 允许http://127.0.0.1:8001域向我发请求 # ret['Access-Control-Allow-Origin']='http://127.0.0.1:8001' # 允许所有人向我发请求 response['Access-Control-Allow-Origin'] = '*' if request.method == 'OPTIONS': # 所有的头信息都允许 response['Access-Control-Allow-Headers'] = '*' return response ``` 在settings中配置即可,在中间件中的位置可以随意放置. 也可以通过第三方组件:pip install django-cors-headers ```python # (1) pip install django-cors-headers # (2) INSTALLED_APPS = ( 'corsheaders', ) # (3) 1 MIDDLEWARE = [ 2 'django.middleware.security.SecurityMiddleware', 3 'django.contrib.sessions.middleware.SessionMiddleware', 4 'django.middleware.csrf.CsrfViewMiddleware', 5 'django.contrib.auth.middleware.AuthenticationMiddleware', 6 'django.contrib.messages.middleware.MessageMiddleware', 7 'django.middleware.clickjacking.XFrameOptionsMiddleware', 8 'corsheaders.middleware.CorsMiddleware', # 按顺序 9 'django.middleware.common.CommonMiddleware', # 按顺序 10 ] # 配置白名单 1 CORS_ALLOW_CREDENTIALS = True#允许携带cookie 2 CORS_ORIGIN_ALLOW_ALL = True 3 CORS_ORIGIN_WHITELIST = ( '*')#跨域增加忽略 4 CORS_ALLOW_METHODS = ( 'DELETE', 'GET', 'OPTIONS', 'PATCH', 'POST', 'PUT', 'VIEW', ) 5 #允许的请求头 6 CORS_ALLOW_HEADERS = ( 7 'XMLHttpRequest', 8 'X_FILENAME', 9 'accept-encoding', 10 'authorization', 11 'content-type', 12 'dnt', 13 'origin', 14 'user-agent', 15 'x-csrftoken', 16 'x-requested-with', 17 'Pragma', 18 ) ``` > 前端项目设置请求头记得添加到CORS_ALLOW_HEADERS # 八、Django的组件 ## 8.1、中间件 中间件顾名思义,是介于request与response处理之间的一道处理过程,相对比较轻量级,并且在全局上改变django的输入与输出。因为改变的是全局,所以需要谨慎实用,用不好会影响到性能。 Django的中间件的定义: ```text Middleware is a framework of hooks into Django’s request/response processing. It’s a light, low-level “plugin” system for globally altering Django’s input or output. MiddleWare,是 Django 请求/响应处理的钩子框架。 它是一个轻量级的、低级的“插件”系统,用于全局改变 Django 的输入或输出。【输入指代的就是客户端像服务端django发送数据,输出指代django根据客户端要求处理数据的结果返回给客户端】 ``` 如果你想修改请求,例如被传送到view中的**HttpRequest**对象。 或者你想修改view返回的**HttpResponse**对象,这些都可以通过中间件来实现。 django框架内部声明了很多的中间件,这些中间件有着各种各种的用途,有些没有被使用,有些被默认开启使用了。而被开启使用的中间件,都是在settngs.py的MIDDLEWARE中注册使用的。 Django默认的`Middleware`: ```python 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', ] ``` ### 8.1.1.自定义中间件 #### (1)定义中间件 创建存放自定义中间件的文件这里选择在app01里创建mdws.py文件: ```python from django.utils.deprecation import MiddlewareMixin class Md1(MiddlewareMixin): def process_request(self, request): print("Md1请求") # return HttpResponse("Md1中断") # 拦截 def process_response(self, request, response): print("Md1返回") return response class Md2(MiddlewareMixin): def process_request(self, request): print("Md2请求") # return HttpResponse("Md2中断") def process_response(self, request, response): print("Md2返回") return response ``` > 1. process_request默认返回None,返回None,则继续执行下一个中间件的process_request;一旦返回响应体对象,则会拦截返回。 > 2. process_response必须有一个形参response,并return response;这是view函数返回的响应体,像接力棒一样传承给最后的客户端。 ![image-20210812132734549](assets/image-20210812132734549.png) #### (2) 注册中间件 ```python MIDDLEWARE = [ ... 'app01.mdws.Md1', 'app01.mdws.Md2' ] ``` #### (3)构建index路由 ```python # path('index/', views.index), def index(request): print("index 视图函数执行...") return HttpResponse("hello yuan") ``` 启动项目,访问index路径: ![image-20210812130256052](assets/image-20210812130256052.png) 后台打印结果: ````text Md1请求 Md2请求 index 视图函数执行... Md2返回 Md1返回 ```` 所以,通过结果我们看出中间件的执行顺序: ![中间件](assets/%E4%B8%AD%E9%97%B4%E4%BB%B6-16287456116091.png) ### 8.1.3.中间件应用 #### 1. 做IP访问频率限制 某些IP访问服务器的频率过高,进行拦截,比如限制每分钟不能超过20次。 ## 8.2、Cookie与Session 我们知道HTTP协议是无状态协议,也就是说每个请求都是独立的!无法记录前一次请求的状态。但HTTP协议中可以使用Cookie来完成会话跟踪!在Web开发中,使用session来完成会话跟踪,session底层依赖Cookie技术。 ### 8.2.1、cookie Cookie翻译成中文是小甜点,小饼干的意思。在HTTP中它表示服务器送给客户端浏览器的小甜点。其实Cookie是key-value结构,类似于一个python中的字典。随着服务器端的响应发送给客户端浏览器。然后客户端浏览器会把Cookie保存起来,当下一次再访问服务器时把Cookie再发送给服务器。 Cookie是由服务器创建,然后通过响应发送给客户端的一个键值对。客户端会保存Cookie,并会标注出Cookie的来源(哪个服务器的Cookie)。当客户端向服务器发出请求时会把所有这个服务器Cookie包含在请求中发送给服务器,这样服务器就可以识别客户端了! > cookie可以理解为每一个浏览器针对每一个服务器创建的key-value结构的本地存储文件 #### (1)cookie流程图 ![image-20210812145309289](assets/image-20210812145309289.png) #### (2)cookie语法 ```python # (1) 设置cookie: res = HttpResponse(...) 或 rep = render(request, ...) 或 rep = redirect() res.set_cookie(key,value,max_age...) res.set_signed_cookie(key,value,salt='加密盐',...)  # (2) 获取cookie: request.COOKIES   # (3) 删除cookie response.delete_cookie("cookie_key",path="/",domain=name) ``` ### 8.2.2、session > Django 提供对匿名会话(session)的完全支持。这个会话框架让你可以存储和取回每个站点访客任意数据。它在服务器端存储数据, 并以cookies的形式进行发送和接受数据。 #### (1)session流程图 ![image-20210813164309357](assets/image-20210813164309357-16288441998871.png) #### (2)session语法与案例 > ```python > # 1、设置Sessions值 > request.session['session_name'] ="admin" > # 2、获取Sessions值 > session_name = request.session["session_name"] > # 3、删除Sessions值 > del request.session["session_name"] > # 4、flush() > # 删除当前的会话数据并删除会话的Cookie。这用于确保前面的会话数据不可以再次被用户的浏览器访问 > ``` ```python def s_login(request): if request.method == "GET": return render(request, "login.html") else: user = request.POST.get("user") pwd = request.POST.get("pwd") try: # user_obj = User.objects.get(user=user,pwd=pwd) # 写session # request.session["is_login"] = True # request.session["username"] = user_obj.user return redirect("/s_index/") except: return redirect("/s_login/") def s_index(request): # 读session is_login = request.session.get("is_login") if is_login: username = request.session.get("username") return render(request, "index.html", {"user": username}) else: return redirect("/s_login/") ''' shop.html: <p> 客户端最后一次访问时间:{{ last_time|default:"第一次访问" }} </p> <h3>商品页面</h3> ''' def shop(request): last_time = request.session.get("last_time") now = datetime.datetime.now().strftime("%Y-%m-%d %X") request.session["last_time"] = now return render(request, "shop.html", {"last_time": last_time}) def s_logout(request): # request.session.flush() del request.session["username"] del request.session["is_login"] return redirect("/s_login/") ``` > 1. session 在服务器端,cookie 在客户端(浏览器) > 2. session 默认被存在在服务器的一个文件里(不是内存) > 3. session 的运行依赖 session id,而 session id 是存在 cookie 中的. > 4. session 可以放在 文件、数据库、或内存中都可以。 > 5. 用户验证这种场合一般会用 session #### (3)session配置 ````python # Django默认支持Session,并且默认是将Session数据存储在数据库中,即:django_session 表中。 # 配置 settings.py SESSION_ENGINE = 'django.contrib.sessions.backends.db' # 引擎(默认) SESSION_COOKIE_NAME = "sessionid" # Session的cookie保存在浏览器上时的key,即:sessionid=随机字符串(默认) SESSION_COOKIE_PATH = "/" # Session的cookie保存的路径(默认) SESSION_COOKIE_DOMAIN = None # Session的cookie保存的域名(默认) SESSION_COOKIE_SECURE = False # 是否Https传输cookie(默认) SESSION_COOKIE_HTTPONLY = True # 是否Session的cookie只支持http传输(默认) SESSION_COOKIE_AGE = 1209600 # Session的cookie失效日期(2周)(默认) SESSION_EXPIRE_AT_BROWSER_CLOSE = False # 是否关闭浏览器使得Session过期(默认) SESSION_SAVE_EVERY_REQUEST = False # 是否每次请求都保存Session,默认修改之后才保存(默认) ```` ### 8.2.3、用户认证组件 Django默认已经提供了认证系统Auth模块,我们认证的时候,会使用auth模块里面给我们提供的表。认证系统包含: - 用户管理 - 权限 - 用户组 - 密码哈希系统 - 用户登录或内容显示的表单和视图 - 一个可插拔的后台系统 admin #### (1)Django用户模型类 Django认证系统中提供了用户模型类User保存用户的数据,默认的User包含以下常见的基本字段: | 字段名 | 字段描述 | | ------------------ | ------------------------------------------------------------ | | `username` | 必选。150个字符以内。 用户名可能包含字母数字,`_`,`@`,`+` `.` 和`-`个字符。 | | `first_name` | 可选(`blank=True`)。 少于等于30个字符。 | | `last_name` | 可选(`blank=True`)。 少于等于30个字符。 | | `email` | 可选(`blank=True`)。 邮箱地址。 | | `password` | 必选。 密码的哈希加密串。 (Django 不保存原始密码)。 原始密码可以无限长而且可以包含任意字符。 | | `groups` | 与`Group` 之间的多对多关系。 | | `user_permissions` | 与`Permission` 之间的多对多关系。 | | `is_staff` | 布尔值。 设置用户是否可以访问Admin 站点。 | | `is_active` | 布尔值。 指示用户的账号是否激活。 它不是用来控制用户是否能够登录,而是描述一种帐号的使用状态。 | | `is_superuser` | 是否是超级用户。超级用户具有所有权限。 | | `last_login` | 用户最后一次登录的时间。 | | `date_joined` | 账户创建的时间。 当账号创建时,默认设置为当前的date/time。 | 上面缺少一些字段,所以后面我们会对当前内置的用户模型进行改造,比如说它里面没有手机号字段,后面我们需要加上。 #### (2)重要方法 Django 用户认证(Auth)组件需要导入 auth 模块 ```python # 认证模块 from django.contrib import auth # 对应数据库用户表,可以继承扩展 from django.contrib.auth.models import User ``` #### (1)用户对象 ```python create() # 创建一个普通用户,密码是明文的。 create_user() # 创建一个普通用户,密码是密文的。 create_superuser() # 与create_user() 相同,但是设置is_staff 和is_superuser 为True。 set_password(*raw_password*) # 设置用户的密码为给定的原始字符串,并负责密码的。 不会保存User对象。当None为raw_password时,密码将设置为一个不可用的密码。 check_password(*raw_password*) # 如果给定的raw_password是用户的真实密码,则返回True,可以在校验用户密码时使用。 ``` #### (2)认证方法 ```python auth.authenticate(username,password) # 将输入的密码转为密文去认证,认证成功返回用户对象,失败则返回None ``` #### (3)登录和注销方法 ```python from django.contrib import auth # 该函数接受一个HttpRequest对象,以及一个认证了的User对象。此函数使用django的session框架给某个已认证的用户附加上session id等信息。 auth.login() # 该函数接受一个HttpRequest对象,无返回值。当调用该函数时,当前请求的session信息会全部清除。该用户即使没有登录,使用该函数也不会报错。 auth.logout() ``` #### (4)request.user ````text Django有一个默认中间件,叫做AuthenticationMiddleware,每次请求进来都会去session中去一个userid,取不到的话,赋值request.user = AnonymousUser() , 一个匿名用户对象。 当用户组件auth.login一旦执行,将userid到session中后,再有请求进入Django,将注册的userid对应的user对象赋值给request.user,即再后面的任何视图函数中都可以从request.user中取到该客户端的登录对象。 ```` #### (5)自定义用户表 > from django.contrib.auth.models import AbstractUser > > 设置Auth认证模块使用的用户模型为我们自己定义的用户模型 > > 格式:“子应用目录名.模型类名” > > AUTH_USER_MODEL = 'users.User' ## 8.3、Django的分页器 > ```python > from django.core.paginator import Paginator > ``` ![image-20210813175020727](assets/image-20210813175020727.png) #### (1) index视图 ```python def index(request): ''' 批量导入数据: Booklist=[] for i in range(100): Booklist.append(Book(title="book"+str(i),price=30+i*i)) Book.objects.bulk_create(Booklist) 分页器的使用: book_list=Book.objects.all() paginator = Paginator(book_list, 10) print("count:",paginator.count) #数据总数 print("num_pages",paginator.num_pages) #总页数 print("page_range",paginator.page_range) #页码的列表 page1=paginator.page(1) # 第1页的page对象 for i in page1: # 遍历第1页的所有数据对象 print(i) print(page1.object_list) #第1页的所有数据 page2=paginator.page(2) print(page2.has_next()) #是否有下一页 print(page2.next_page_number()) #下一页的页码 print(page2.has_previous()) #是否有上一页 print(page2.previous_page_number()) #上一页的页码 # 抛错 #page=paginator.page(12) # error:EmptyPage #page=paginator.page("z") # error:PageNotAnInteger ''' book_list = Book.objects.all() paginator = Paginator(book_list, 10) page = request.GET.get('page', 1) current_page = int(page) try: print(page) book_list = paginator.page(page) except PageNotAnInteger: book_list = paginator.page(1) except EmptyPage: book_list = paginator.page(paginator.num_pages) return render(request, "index.html", {"book_list": book_list, "paginator": paginator, "currentPage": current_page}) ``` #### (2) index.html ```html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> </head> <body> <div class="container"> <h4>分页器</h4> <ul> {% for book in book_list %} <li>{{ book.title }} -----{{ book.price }}</li> {% endfor %} </ul> <ul class="pagination" id="pager"> {% if book_list.has_previous %} <li class="previous"><a href="/index/?page={{ book_list.previous_page_number }}">上一页</a></li> {% else %} <li class="previous disabled"><a href="#">上一页</a></li> {% endif %} {% for num in paginator.page_range %} {% if num == currentPage %} <li class="item active"><a href="/index/?page={{ num }}">{{ num }}</a></li> {% else %} <li class="item"><a href="/index/?page={{ num }}">{{ num }}</a></li> {% endif %} {% endfor %} {% if book_list.has_next %} <li class="next"><a href="/index/?page={{ book_list.next_page_number }}">下一页</a></li> {% else %} <li class="next disabled"><a href="#">下一页</a></li> {% endif %} </ul> </div> </body> </html> ``` ## 8.4、FBV与CBV > 1 FBV :function based view > > 2 BCV:class based view ### 8.4.1、前后端分离模式 在开发Web应用中,有两种应用模式: 1. 前后端不分离[客户端看到的内容和所有界面效果都是由服务端提供出来的。] ![前后端不分离](assets/depended_frontend_backend.png) 2. 前后端分离【把前端的界面效果(html,css,js分离到另一个服务端,python服务端只需要返回数据即可)】 前端形成一个独立的网站,服务端构成一个独立的网站 ![](assets/indepent_frontend_backend-16318493319127.png)