博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
python 全栈开发,Day67(Django简介)
阅读量:5328 次
发布时间:2019-06-14

本文共 13426 字,大约阅读时间需要 44 分钟。

昨日内容回顾

1. socket创建服务器2. http协议:    请求协议       请求首行     请求方式 url?a=1&b=2  协议       请求头       key:value       请求体       a=1&b=2(只有post请求才有请求体)             响应协议         响应首行    协议 状态码 文本        响应头      key:value        响应体      html字符串3. wsgiref模块(基于wsgi协议)    功能:        1. 按着http协议请求格式解析请求数据----envision:{}                2. 按着http协议响应格式封装响应数据----response            4 基于wsgiref实现了一个简单web框架    1. urls : 存放路由关系    2 views: 存放视图函数    3 templates: 存放html文件    4 wsgi-sever:启动文件

 

一、Django简介

知识预览

  • MVC与MTV模型

  • Django的下载与基本命令

  • 基于Django实现的一个简单示例

 

MVC与MTV模型

MVC

Web服务器开发领域里著名的MVC模式,所谓MVC就是把Web应用分为模型(M),控制器(C)和视图(V)三层,他们之间以一种插件式的、松耦合的方式连接在一起,模型负责业务对象与数据库的映射(ORM),视图负责与用户的交互(页面),控制器接受用户的输入调用模型和视图完成用户的请求,其示意图如下所示:

mvc主要用于web框架,常用的开发语言,有java,php,node.js等等。

web框架应用最广泛就是PHP了,它只能做web开发,而且开发效率很快。

 

MTV

Django的MTV模式本质上和MVC是一样的,也是为了各组件间保持松耦合关系,只是定义上有些许不同,Django的MTV分别是值:

M 代表模型(Model): 负责业务对象和数据库的关系映射(ORM)。

T 代表模板 (Template):负责如何把页面展示给用户(html)。
V 代表视图(View): 负责业务逻辑,并在适当时候调用Model和Template。
除了以上三层之外,还需要一个URL分发器,它的作用是将一个个URL的页面请求分发给不同的View处理,View再调用相应的Model和Template,MTV的响应模式如下所示:

一般是用户通过浏览器向我们的服务器发起一个请求(request),这个请求回去访问视图函数,(如果不涉及到数据调用,那么这个时候视图函数返回一个模板也就是一个网页给用户),视图函数调用模型,模型去数据库查找数据,然后逐级返回,视图函数把返回的数据填充到模板中空格中,最后返回网页给用户。

 

这里面最难的部分就是model,后面会慢慢讲到。

django 有一个ORM,它是专门来操作数据库的。这套语法,需要大量练习才能掌握。

 

MVC和MTV模型的区别:

MVC:    M : model (与数据库打交道)    V :  views  (存放html文件)    C : Controller(逻辑控制部分)        MTV     M : model     (与数据库打交道)    T : templates    (存放html文件)        V : views      (逻辑处理)    +    路由控制层(分发哪一个路径由哪一个视图函数处理),它没有单独的分层。它作为URL分发器,将url请求分发给不同的view处理

 

Django的下载与基本命令

1、下载Django:

pip3 install django

2、创建一个django project

windows用户,以管理员身份打开一个cmd窗口。进入一个空目录,运行以下命令:

E:\python_script\django框架\day2>django-admin startproject mysite

当前目录下会生成mysite的工程,目录结构如下:

mysite/├── manage.py└── mysite    ├── __init__.py    ├── settings.py    ├── urls.py    └── wsgi.py

manage.py ----- Django项目里面的工具,通过它可以调用django shell和数据库等。

settings.py ---- 包含了项目的默认设置,包括数据库信息,调试标志以及其他一些工作的变量。
urls.py ----- 负责把URL模式映射到应用程序。

 

manage.py : 它不关是启动文件,它还是与Django交互的文件。比如:

python manage.py  runserver   : 运行项目python manage.py  startapp    : 创建应用

如果运行项目时,不指定端口,默认监听本机的8000端口。

3、在mysite目录下创建应用

#进入mysite目录E:\python_script\django框架\day2>cd mysite#创建应用blogE:\python_script\django框架\day2\mysite>python manage.py startapp blog

目录结构如下:

mysite/├── blog│   ├── admin.py│   ├── apps.py│   ├── __init__.py│   ├── migrations│   │   └── __init__.py│   ├── models.py│   ├── tests.py│   └── views.py├── manage.py└── mysite    ├── __init__.py    ├── settings.py    ├── urls.py    └── wsgi.py

views.py---存放视图函数

models--与数据库打交道

还有一个目录templates,它是用来存放html文件的,下面会讲到。

 

从上面的目录结构可以看出,mysite目录下有一个blog。那么顶层的mysite,叫做 项目。底层的blog叫做应用。

比如微信是一个项目。聊天,朋友圈,支付...都是应用。

项目是必须包含应用的,项目可以包含多个应用。

mysite下的mysite,是全局文件,它有2个全局配置文件,一个是settings.py(项目配置文件),一个是urls.py(路由控制文件)。

wsgi.py是封装socket,用来接收和响应请求的。这个文件,从来都不需要动。

 

4、启动django项目

E:\python_script\django框架\day2\mysite>python manage.py runserver 8080Performing system checks...System check identified no issues (0 silenced).You have 14 unapplied migration(s). Your project may not work properly until you apply the migrations for app(s): admin, auth, contenttypes, sessions.Run 'python manage.py migrate' to apply them.June 21, 2018 - 19:33:29Django version 2.0.6, using settings 'mysite.settings'Starting development server at http://127.0.0.1:8080/Quit the server with CTRL-BREAK.

这样我们的django就启动起来了!当我们访问:http://127.0.0.1:8080/时就可以看到:

基于Django实现的一个简单示例

url控制器

修改mysite目录下的urls.py,增加index路径

注意:index后面不要加括号。直接views.index即可

必须导入blog应用的views模块,否则它找不到对应的视图函数

from django.contrib import adminfrom django.urls import pathfrom blog import viewsurlpatterns = [    path('admin/', admin.site.urls),    path('index/',views.index),]
View Code

视图

修改blog目录下的views.py,增加index视图函数

from django.shortcuts import renderimport datetime# Create your views here.def index(request):    now=datetime.datetime.now()    ctime=now.strftime("%Y-%m-%d %X")    return render(request,"index.html",{
"ctime":ctime})
View Code

request,它是一个对象。存储了请求信息,比如请求路径,请求方式,GET数据,POST数据...等等。

request参数必须要有,不管你用不用它。

 

模板

新建文件夹templates,在此目录创建index.html

    
Title

当前时间:{
{ ctime }}

View Code

修改mysite目录下的settings.py,指定模板目录为templates,修改部分如下:

TEMPLATES = [    {        'BACKEND': 'django.template.backends.django.DjangoTemplates',        'DIRS': [os.path.join(BASE_DIR, 'templates')],        'APP_DIRS': True,        'OPTIONS': {            'context_processors': [                'django.template.context_processors.debug',                'django.template.context_processors.request',                'django.contrib.auth.context_processors.auth',                'django.contrib.messages.context_processors.messages',            ],        },    },]
View Code

访问网页,效果如下:

 django有一个好处,代码更改之后,它会自动加载代码。而不需要重启django项目,网页就能更新了!

 

增加登录页面

修改mysite目录下的urls.py,新增一个login

urlpatterns = [    path('admin/', admin.site.urls),    path('index/',views.index),    path('login/',views.login),]
View Code

在templates目录下创建文件login.html

注意:form表单的标签名是form,不是from。from是MySQL的关键字,不要弄混淆了。否则点击提交按钮,是没有反应的。

    
Title
用户名
用户名
View Code

修改blog目录下的views.py,增加login视图函数

from django.shortcuts import renderimport datetime# Create your views here.def index(request):    now=datetime.datetime.now()    ctime=now.strftime("%Y-%m-%d %X")    return render(request,"index.html",{
"ctime":ctime})def login(request): return render(request,"login.html")
View Code

访问登录页面,效果如下:

为什么render能找到login.html文件呢?

因为setting.py文件里面定义了template路径。render方法,是用来渲染模板的,它会从TEMPLATES配置的路径中去寻找html文件。

如果修改DIRS里面的文件名,比如改为abc

'DIRS': [os.path.join(BASE_DIR, 'abc')],

访问页面,会报错

重新修改回来,再次访问,就正常了。

 

修改urls.py,增加auth路径,用来做验证的。

urlpatterns = [    path('admin/', admin.site.urls),    path('index/',views.index),    path('login/',views.login),    path('auth/',views.auth),]
View Code

修改login.html文件,改为post请求

    
Title
用户名
用户名
View Code

修改views.py文件,增加auth视图函数

from django.shortcuts import render,HttpResponseimport datetime# Create your views here.def index(request):    now=datetime.datetime.now()    ctime=now.strftime("%Y-%m-%d %X")    return render(request,"index.html",{
"ctime":ctime})def login(request): return render(request,"login.html")def auth(request): print(request.path) # 路径 print(request.method) # 请求方式 print(request.GET) # GET数据 print(request.POST) # POST数据 return HttpResponse("OK")
View Code

访问登录页面,输入数据,点击提交

页面输出403,被CSRF拦截了。

CSRF:跨站请求伪造,常缩写为CSRF或者XSRF,是一种对网站的恶意利用。

后面的课程会讲到,如何避免CSRF。修改settings.py里面的MIDDLEWARE 配置项,关闭CSRF

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',]
View Code

 

访问方式

访问方式有2种,GET和POST

在地址栏中,只有GET请求。

在form表单中,有GET和POST。它是根据method属性决定的!一般表单使用POST

 

再次访问url,输入表单信息,点击提交。

输出ok,表示正常。注意:这里还没有做登录认证,下面会讲到!

查看cmd窗口输出信息:

/auth/POST

可以看到:

路径:/auth/。请求方式: POST。GET数据为空。POST数据是一个字典。

 

地址栏直接输入:

http://127.0.0.1:8080/auth/?u=xiao,sex=m

查看cmd窗口输出信息:

/auth/GET

 

登录认证

正常网站,用户名和密码是保存在数据库中。由于还没有学到django连接数据库,所以这里将用户名和密码写死,模拟登录行为。

修改views.py,获取用户和密码,进行判断。

from django.shortcuts import render,HttpResponseimport datetime# Create your views here.def index(request):    now=datetime.datetime.now()    ctime=now.strftime("%Y-%m-%d %X")    return render(request,"index.html",{
"ctime":ctime})def login(request): return render(request,"login.html")def auth(request): user = request.POST.get('user') # 获取用户名 pwd = request.POST.get('pwd') # 获取密码 print(user,pwd) #判断用户名和密码 if user == 'xiao' and pwd == '123': return HttpResponse("登录成功") # 返回响应体给浏览器,显示"登录成功"文字 else: return render(request,"login.html") # 返回响应体-->login.html文件内容
View Code

重新访问登录页面,输入正确的用户名和密码

页面提示,成功。

 

访问过程分析

访问登录页面时,经历3个步骤

(1) http://127.0.0.1:8000/login/   get请求 无数据(2) path('login/',views.login),    调转视图函数login(request)(3) login                          执行视图函数,响应了一个login.html页面

解释:

1. 首先是用户在浏览器输入url:http://127.0.0.1:8000/login/

2. django接收到请求之后,根据URL控制器匹配视图函数

3. 执行视图函数login,响应请求给浏览器一个login.html页面。

 

查看views.py文件的login视图函数

render(request,"login.html")

上面的代码就是响应体。那么浏览器如何得到response信息的呢?封装response信息是由wsgi来完成的。

 

点击提交按钮的操作,也经历3个步骤

(1) http://127.0.0.1:8000/auth/   post请求,数据为user=xiao&pwd=123(2) path('auth/',views.auth),    调取视图函数auth(request)(3) auth      执行视图函数, if 登陆成功:响应一个字符串登陆成功。else:  响应了一个登陆页面

解释:

1. 虽然form的action属性值为"/auth/",但是执行提交动作时,浏览器会查看action属性,如果为相对路径。那么会获取当前url的域名/IP加端口。和action属性拼接,得到完整的url,比如:http://127.0.0.1:8000/auth/。将表单数据以POST方式发送给此url。

注意:推荐使用这种写法。如果action写成完整的url(比如:http://127.0.0.1:8000/auth/),遇到服务器迁移时。那么涉及到的html文件,都需要更改,非常耗时耗力!

如果采用相对路径方式,那么不需要改动代码,它会自动拼接,完美解决这个问题。

比如写/auth/,会自动拼接为http://127.0.0.1:8000/auth/

如果action为"",也就是空,它会拼接当前的完整ur。

比如访问登录页面,那么action的属性值为 当前url,比如:http://127.0.0.1:8000/login/

2. django接收到请求之后,根据URL控制器匹配视图函数auth

3. 执行视图函数,如果用户名和密码正确,页面显示登录成功。否则,页面还是显示登录页面。

 

上面提到的2个场景,它们之间,是没有任何关系的。

每一个请求,对于服务器而言,都是一个新的请求。

 

思考一个问题,能够将login和auth视图函数合并?

答案是可以的。

更改login.html,将action属性设置为空(参考上面的步骤1解释)

更改views.py,删除auth视图函数代码,修改login视图函数,完整代码如下:

from django.shortcuts import render,HttpResponseimport datetime# Create your views here.def index(request):    now=datetime.datetime.now()    ctime=now.strftime("%Y-%m-%d %X")    return render(request,"index.html",{
"ctime":ctime})def login(request): #判断请求是否为POST,必须为大写 if request.method == "POST": user = request.POST.get('user') # 获取用户名 pwd = request.POST.get('pwd') # 获取密码 print(user, pwd) # 判断用户名和密码 if user == 'xiao' and pwd == '123': return HttpResponse("登录成功") # 返回响应体给浏览器,显示"登录成功"文字 else: return render(request, "login.html") # 返回响应体-->login.html文件内容 return render(request,"login.html") # 默认输出登录页面
View Code

修改urls.py,删除auth路径

urlpatterns = [    path('admin/', admin.site.urls),    path('index/',views.index),    path('login/',views.login),]
View Code

重新访问登录页面,输入正确的用户和密码,点击提交。页面输出:

这就用到了if分支。

能尽量合成视图函数的,推荐合成。如果逻辑简单,可以合成。
逻辑比较复杂的,还是建议分开。

 

视图函数,必须返回一个HttpResponse对象。HttpResponse是一个对象,对象里面,放字符串。

HttpResponse会自动将字符串转换为字节

django要求视图函数,必须返回一个HttpResponse对象。

 

模拟render操作

修改login函数,else部分是重点

def login(request):    #判断请求是否为POST,必须为大写    if request.method == "POST":        user = request.POST.get('user')  # 获取用户名        pwd = request.POST.get('pwd')  # 获取密码        print(user, pwd)        # 判断用户名和密码        if user == 'xiao' and pwd == '123':            return HttpResponse("登录成功")  # 返回响应体给浏览器,显示"登录成功"文字        else:            from mysite import settings  # 导入settings模块            import os            # 拼接login.html的绝对路径            path = os.path.join(settings.BASE_DIR,"templates","login.html")            with open(path,encoding="utf-8") as f:                data = f.read()  # 读取文件所有内容            print("data",data+'aaaaa')            #返回给浏览器并加上一段话            return HttpResponse(data+'用户名和密码错误')            # return render(request, "login.html")  # 返回响应体-->login.html文件内容    return render(request,"login.html")  # 默认输出登录页面
View Code

访问url: http://127.0.0.1:8000/login/

输入一个错误的密码,点击提交

页面输出,用户名和密码错误

那么,render就是干了这些事情。

 

总结:

对于Django而言,一次请求必须返回一个HttpResponse(字符串)

request对象,存放了请求路径,请求方式,请求数据,比如GET和POST

所以对于视图函数而言,最关心的部分就是request和HttpResponse

一次请求,必有一次响应。如果没有响应,就会报错

 

范围url: http://127.0.0.1:8000/index/

在视图函数中,render是渲染的意思。那么它是如何工作的呢?

1 按着settings-TEMPLATES-DIRS路径找指定文件2 读取文件所有字符串3 渲染: 检查字符串中是否有{
{变量}} , if 没有找到: HttpResponse(文件字符串) else 找到 {
{变量}},用render第三个参数中的对应值进行相应替换(如果没有找到对应值,{
{变量}}替换为空) HttpResponse(替换后的文件字符串)

那么渲染的过程,是在后端完成的。不是前端完成的。

看html代码,就知道了。浏览器根本不认识{
{变量}},它只能识别html,css,js

注意:如果模板里面,写了{

{变量}} 。但是render没传,那么页面中{
{变量}} 会被替换为空。

如果模板里面,写了{

{ }} 。变量名没写,那么页面报错

 

如果render传了变量,但是模板里{

{变量}} ,变量名写错了,页面中{
{变量}} 也会被替换为空。

 

 

思考:如何点击时间的时候,变成红色?

 直接加行内样式?不对,它是点击的时候,才变成红色。

需要引入jquery来做,修改index.html

    
Title

当前时间: {
{ ctime }}

View Code

访问url: http://127.0.0.1:8000/index/

点击时间,就会变红

但是,线上服务器不能这么干?为什么呢?因为如果一旦jquery访问链接失效。那么整个网站就崩溃了!

所以这种文件,还是需要放到自己的服务器上,才行!

那好办呀,将jquery.min.js放到templates目录。

编辑index.html,直接引入jquery.min.js文件。

再次访问页面,怎么点击都没效果,查看控制台,点击网络部分,发现它是404了!

不要以为templates下的文件,可以随便访问。太天真了!

浏览器是不能直接访问templates下的文件,需要Django找到静态文件才行!

在根目录,创建static目录,它是专门存放静态文件的。

将js文件进去。项目目录结构如下:

mysite/├── blog│   ├── admin.py│   ├── apps.py│   ├── __init__.py│   ├── models.py│   ├── tests.py│   └── views.py├── manage.py├── mysite│   ├── __init__.py│   ├── settings.py│   ├── urls.py│   └── wsgi.py├── static│   └── jquery.min.js└── templates    ├── index.html    └── login.html

修改settings.py,最后一行添加,注意:STATIC_URL和它是成对使用的。

STATIC_URL = '/static/'STATICFILES_DIRS = (    os.path.join(BASE_DIR,"static"),)

STATIC_URL参数,表示别名。

STATICFILES_DIRS表示物理路径。

STATIC_URL代指STATICFILES_DIRS定义的路径。

修改index.html,更改src属性

注意:这里面的/static/ 是别名,它代指的是物理路径

重新访问页面,再次点击,就会变红。

 

因为diango利用前缀STATIC_URL的具体内容,来映射STATICFILES_DIRS, 那么它就可以找到具体的文件。

比如前台页面的静态资源路径,一般都是写死了,可能涉及到几百个网页。网站在运营过程中,难免后台服务器,需要做迁移工作,可能和之前的存储路径不一样的。这个时候,让前端去改几百个网页,是一个很繁杂的工作。现在只需要修改STATIC_URL,就可以完美解决这个问题!!!

 

 未完待续...

 

在 success: 后面定义回调函数处理返回的数据,需要使用 JSON.parse(data)


参考资料:

转载声明:

作者:
出处: 

转载于:https://www.cnblogs.com/bqwzx/p/10158813.html

你可能感兴趣的文章
WPF中Image显示本地图片
查看>>
Hyper-V虚拟机上安装一个图形界面的Linux系统
查看>>
js千分位处理
查看>>
字符串类型的相互转换
查看>>
基础学习:C#中float的取值范围和精度
查看>>
Vim配置Node.js开发工具
查看>>
web前端面试题2017
查看>>
ELMAH——可插拔错误日志工具
查看>>
MySQL学习笔记(四)
查看>>
【Crash Course Psychology】2. Research & Experimentation笔记
查看>>
SOPC Builder中SystemID
查看>>
关于 linux 的 limit 的设置
查看>>
HDU(4528),BFS,2013腾讯编程马拉松初赛第五场(3月25日)
查看>>
vim中文帮助教程
查看>>
MySQL基础3
查看>>
RxJS & Angular
查看>>
面向对象(多异常的声明与处理)
查看>>
MTK笔记
查看>>
ERROR: duplicate key value violates unique constraint "xxx"
查看>>
激活office 365 的启动文件
查看>>