day052-53 django框架
一、MVC和MTV模型
这就是web开发中的一种思维模式或者说一套理念,MTV也是基于MVC发展出来的,本质相同,都是使各组件保持松耦合
MVC
把web应用分为模型(model),控制器(controller),视图(view)三层
视图负责和用户交互,模型负责业务对象和数据库之间的映射,控制器接收用户的输入调用模型和视图完成用户的请求。如图
MTV
m代表模型(model):负责业务对象和数据库的关系映射
T代表模板(template):负责如何把页面展示给用户
v代表视图(view):负责业务逻辑,并在适当的时候调用model和template
最后,还需要一个url分发器,它负责把相应的url页面的请求发给不同的view,然后view再调用model和template
一般的流程是:
用户发起一个请求,这个请求根据URL控制器的分配,访问相应的视图函数,视图函数到model里取数据,并把取到的数据传给template,template把数据渲染到相应的位置,返回html给用户浏览器,浏览器展示。如图
二、Django的基本命令
#创建项目
django-admin.py startproject 项目名
#在项目目录下创建应用
python manage.py startapp 应用名
#启动项目
pyhton manage.py runserver ip:端口(不写的话默认是127.0.0.1:8000)
三、Django的路由层(URLconf)
url配置(URLconf)就像网站的目录。本质是url和该URL要调用的而视图函数之间的映射表。
1 from django.conf.urls import url 2 3 from . import views 4 5 urlpatterns = [ 6 url(r'^articles/2003/$', views.special_case_2003), 7 url(r'^articles/([0-9]{4})/$', views.year_archive), 8 url(r'^articles/([0-9]{4})/([0-9]{2})/$', views.month_archive), 9 url(r'^articles/([0-9]{4})/([0-9]{2})/([0-9]+)/$', views.article_detail), 10 ]
注意:
#、url匹配用的是正则表达式,所以,要想从url获取一个值,只需要在它周围加一个括号。
#、不需要添加前导的反斜杠,因为每个url都有,例如,应该是^articles 而不是^articles
#、正则表达式最好在前边加一个“ r ”,取消转义
2、有名分组
可以使用命名的正则表达式组来捕获url中的值并以关键字参数传递给视图函数
没有命名的正则表达式组捕获的值只能通过位置参数来传递给视图函数
1 from django.conf.urls import url 2 3 from . import views 4 5 urlpatterns = [ 6 url(r'^articles/2003/$', views.special_case_2003), 7 url(r'^articles/(?P<year>[0-9]{4})/$', views.year_archive), 8 url(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$', views.month_archive), 9 url(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/(?P<day>[0-9]{2})/$', views.article_detail), 10 ]
这样的好处是不会产生参数顺序错误的问题
3、分发
将url控制器再细分成不同的层级,比如每个应用也有它自己的url控制器,主url控制器收到请求后再分发给相应的子url控制器,这样可以降低主url控制器的负担。
1 from django.conf.urls import include, url 2 3 urlpatterns = [ 4 url(r'^admin/', admin.site.urls), 5 url(r'^blog/', include('blog.urls')), 6 ]
4、反向解析
就是在url配置中给每个url在起一个别名,在用到这个url的地方就用这个别名代替。这样避免url硬编码,提高扩展性。
在需要url的地方,对于不同的层级,Django提供不同的工具用于url反向解析:
在模板中,使用url模板标签
在python代码中,使用django.core.urlresolvers.reverse()函数。
from django.conf.urls import url from . import views urlpatterns = [ #... url(r'^articles/([0-9]{4})/$', views.year_archive, name='news-year-archive'), #... ]
1 <a href="{% url 'news-year-archive' 2012 %}">2012 Archive</a> 2 3 <ul> 4 {% for yearvar in year_list %} 5 <li><a href="{% url 'news-year-archive' yearvar %}">{{ yearvar }} Archive</a></li> 6 {% endfor %} 7 </ul>
1 #使用reverse() 2 3 from django.core.urlresolvers import reverse 4 from django.http import HttpResponseRedirect 5 6 def redirect_to_year(request): 7 # ... 8 year = 2006 9 # ... 10 return HttpResponseRedirect(reverse('news-year-archive', args=(year,))) # 同redirect("/path/")
四、Django的视图层
视图指的就是视图函数,它接收web请求并返回web响应,响应可以是y一个网页html,一个重定向,一个404错误,一个xml文档,或者一张图片,可以是任何东西,但都是字符串的形式,视图可以放在任何地方,但一般放在views.py中。
两个重要对象:
1、HTTPRequest对象
django将请求报文中的请求行、首部信息、内容主体封装成 HttpRequest 类中的属性。 除了特殊说明的之外,其他均为只读的。
1 1.HttpRequest.GET 2 3 一个类似于字典的对象,包含 HTTP GET 的所有参数。详情请参考 QueryDict 对象。 4 5 2.HttpRequest.POST 6 7 一个类似于字典的对象,如果请求中包含表单数据,则将这些数据封装成 QueryDict 对象。 8 9 POST 请求可以带有空的 POST 字典 —— 如果通过 HTTP POST 方法发送一个表单,但是表单中没有任何的数据,QueryDict 对象依然会被创建。 10 因此,不应该使用 if request.POST 来检查使用的是否是POST 方法;应该使用 if request.method == "POST" 11 另外:如果使用 POST 上传文件的话,文件信息将包含在 FILES 属性中。 12 13 注意:键值对的值是多个的时候,比如checkbox类型的input标签,select标签,需要用: 14 request.POST.getlist("hobby") 15 16 3.HttpRequest.body 17 18 一个字符串,代表请求报文的主体。在处理非 HTTP 形式的报文时非常有用,例如:二进制图片、XML,Json等。 19 但是,如果要处理表单数据的时候,推荐还是使用 HttpRequest.POST 。 20 21 22 4.HttpRequest.path 23 24 一个字符串,表示请求的路径组件(不含域名)。 25 例如:"/music/bands/the_beatles/" 26 27 5.HttpRequest.method 28 29 一个字符串,表示请求使用的HTTP 方法。必须使用大写。 30 例如:"GET"、"POST" 31 32 33 6.HttpRequest.encoding 34 35 一个字符串,表示提交的数据的编码方式(如果为 None 则表示使用 DEFAULT_CHARSET 的设置,默认为 'utf-8')。 36 这个属性是可写的,你可以修改它来修改访问表单数据使用的编码。 37 接下来对属性的任何访问(例如从 GET 或 POST 中读取数据)将使用新的 encoding 值。 38 如果你知道表单数据的编码不是 DEFAULT_CHARSET ,则使用它。 39 40 41 7.HttpRequest.META 42 43 一个标准的Python 字典,包含所有的HTTP 首部。具体的头部信息取决于客户端和服务器,下面是一些示例: 44 45 CONTENT_LENGTH —— 请求的正文的长度(是一个字符串)。 46 CONTENT_TYPE —— 请求的正文的MIME 类型。 47 HTTP_ACCEPT —— 响应可接收的Content-Type。 48 HTTP_ACCEPT_ENCODING —— 响应可接收的编码。 49 HTTP_ACCEPT_LANGUAGE —— 响应可接收的语言。 50 HTTP_HOST —— 客服端发送的HTTP Host 头部。 51 HTTP_REFERER —— Referring 页面。 52 HTTP_USER_AGENT —— 客户端的user-agent 字符串。 53 QUERY_STRING —— 单个字符串形式的查询字符串(未解析过的形式)。 54 REMOTE_ADDR —— 客户端的IP 地址。 55 REMOTE_HOST —— 客户端的主机名。 56 REMOTE_USER —— 服务器认证后的用户。 57 REQUEST_METHOD —— 一个字符串,例如"GET" 或"POST"。 58 SERVER_NAME —— 服务器的主机名。 59 SERVER_PORT —— 服务器的端口(是一个字符串)。 60 从上面可以看到,除 CONTENT_LENGTH 和 CONTENT_TYPE 之外,请求中的任何 HTTP 首部转换为 META 的键时, 61 都会将所有字母大写并将连接符替换为下划线最后加上 HTTP_ 前缀。 62 所以,一个叫做 X-Bender 的头部将转换成 META 中的 HTTP_X_BENDER 键。 63 64 8.HttpRequest.FILES 65 66 一个类似于字典的对象,包含所有的上传文件信息。 67 FILES 中的每个键为<input type="file" name="" /> 中的name,值则为对应的数据。 68 注意,FILES 只有在请求的方法为POST 且提交的<form> 带有enctype="multipart/form-data" 的情况下才会 69 包含数据。否则,FILES 将为一个空的类似于字典的对象。 70 71 72 9.HttpRequest.COOKIES 73 74 一个标准的Python 字典,包含所有的cookie。键和值都为字符串。 75 76 77 78 10.HttpRequest.session 79 80 一个既可读又可写的类似于字典的对象,表示当前的会话。只有当Django 启用会话的支持时才可用。 81 完整的细节参见会话的文档。 82 83 84 11.HttpRequest.user(用户认证组件下使用) 85 86 一个 AUTH_USER_MODEL 类型的对象,表示当前登录的用户。 87 88 如果用户当前没有登录,user 将设置为 django.contrib.auth.models.AnonymousUser 的一个实例。你可以通过 is_authenticated() 区分它们。 89 90 例如: 91 92 if request.user.is_authenticated(): 93 # Do something for logged-in users. 94 else: 95 # Do something for anonymous users. 96 97 98 user 只有当Django 启用 AuthenticationMiddleware 中间件时才可用。 99 100 ------------------------------------------------------------------------------------- 101 102 匿名用户 103 class models.AnonymousUser 104 105 django.contrib.auth.models.AnonymousUser 类实现了django.contrib.auth.models.User 接口,但具有下面几个不同点: 106 107 id 永远为None。 108 username 永远为空字符串。 109 get_username() 永远返回空字符串。 110 is_staff 和 is_superuser 永远为False。 111 is_active 永远为 False。 112 groups 和 user_permissions 永远为空。 113 is_anonymous() 返回True 而不是False。 114 is_authenticated() 返回False 而不是True。 115 set_password()、check_password()、save() 和delete() 引发 NotImplementedError。 116 New in Django 1.8: 117 新增 AnonymousUser.get_username() 以更好地模拟 django.contrib.auth.models.User。 118 119 */
1 1.HttpRequest.get_full_path() 2 3 返回 path,如果可以将加上查询字符串。 4 5 例如:"/music/bands/the_beatles/?print=true" 6 7 8 2.HttpRequest.is_ajax() 9 10 如果请求是通过XMLHttpRequest 发起的,则返回True,方法是检查 HTTP_X_REQUESTED_WITH 相应的首部是否是字符串'XMLHttpRequest'。 11 12 大部分现代的 JavaScript 库都会发送这个头部。如果你编写自己的 XMLHttpRequest 调用(在浏览器端),你必须手工设置这个值来让 is_ajax() 可以工作。 13 14 如果一个响应需要根据请求是否是通过AJAX 发起的,并且你正在使用某种形式的缓存例如Django 的 cache middleware, 15 你应该使用 vary_on_headers('HTTP_X_REQUESTED_WITH') 装饰你的视图以让响应能够正确地缓存。
2、HTTPResponse对象
响应主要有是那种方法HTTPResponse、render、redirect
#HTTPResponse方法
HttpResponse()括号内直接跟一个具体的字符串作为响应体
#render方法
render(request, template_name,[ context])
参数: request: 用于生成响应的请求对象。 template_name:要使用的模板的完整名称,可选的参数 context:添加到模板上下文的一个字典。默认是一个空字典。如果字典中的某个值是可调用的,视图将在渲染模板之前调用它。
render方法就是将一个模板页面中的模板语法进行渲染,最终渲染成一个html页面作为响应体。
1 good_list=['小米电视',"海尔冰箱","格力空调","AAA"] 2 return render(request,"index.html",{"good_list":good_list}) 3 4
3、redirect方法
传递一个重定向的url
1 def my_view(request): 2 ... 3 return redirect('/some/url/') 4 5 #也可以是完整的url 6 def my_view(request): 7 ... 8 return redirect('http://example.com/')
重定向过程中浏览器有两次请求,第一次请求收到一个301或302的状态码的响应,包含了重定向的地址,第二次是向这个地址发起请求。
301和302的区别
1 1)301和302的区别。 2 3 301和302状态码都表示重定向,就是说浏览器在拿到服务器返回的这个状态码后会自动跳转到一个新的URL地址,这个地址可以从响应的Location首部中获取 4 (用户看到的效果就是他输入的地址A瞬间变成了另一个地址B)——这是它们的共同点。 5 6 他们的不同在于。301表示旧地址A的资源已经被永久地移除了(这个资源不可访问了),搜索引擎在抓取新内容的同时也将旧的网址交换为重定向之后的网址; 7 8 302表示旧地址A的资源还在(仍然可以访问),这个重定向只是临时地从旧地址A跳转到地址B,搜索引擎会抓取新的内容而保存旧的网址。 SEO302好于301 9 10 11 12 2)重定向原因: 13 (1)网站调整(如改变网页目录结构); 14 (2)网页被移到一个新地址; 15 (3)网页扩展名改变(如应用需要把.php改成.Html或.shtml)。 16 这种情况下,如果不做重定向,则用户收藏夹或搜索引擎数据库中旧地址只能让访问客户得到一个404页面错误信息,访问流量白白丧失;再者某些注册了多个域名的网站,也需要通过重定向让访问这些域名的用户自动跳转到主站点等。
五、Django模板层
Django的模板系统:HTML+模板语法
1、模板语法之变量
{{ var_name }}
3、模板语法之过滤器
{{ obj | filter_name:param }}
3、模板语法之标签
{% tag %}
4、自定义标签和过滤器
5、模板继承
模版继承可以让您创建一个基本的“骨架”模版,它包含您站点中的全部元素,并且可以定义能够被子模版覆盖的 blocks 。