免费网站制作下载,东莞网站建设模具,智慧团建官网登录口,中山外贸网站开发http请求中产生两个核心对象#xff1a; http请求#xff1a;HttpRequest对象 http响应#xff1a;HttpResponse对象 所在位置#xff1a;django.http 之前我们用到的参数request就是HttpRequest 检测方法#xff1a;isinstance(request,HttpRequest) 1 HttpRequest对象… http请求中产生两个核心对象 http请求HttpRequest对象 http响应HttpResponse对象 所在位置django.http 之前我们用到的参数request就是HttpRequest 检测方法isinstance(request,HttpRequest) 1 HttpRequest对象的属性和方法 # path 请求页面的全路径不包括域名
#
# method 请求中使用的HTTP方法的字符串表示。全大写表示。例如
#
# if req.methodGET:
#
# do_something()
#
# elseif req.methodPOST:
#
# do_something_else()
#
# GET: 包含所有HTTP GET参数的类字典对象
#
# POST 包含所有HTTP POST参数的类字典对象
#
# 服务器收到空的POST请求的情况也是可能发生的也就是说表单form通过
# HTTP POST方法提交请求但是表单中可能没有数据因此不能使用
# if req.POST来判断是否使用了HTTP POST 方法应该使用 if req.methodPOST
#
#
#
# COOKIES: 包含所有cookies的标准Python字典对象keys和values都是字符串。
#
# FILES 包含所有上传文件的类字典对象FILES中的每一个Key都是input typefile name /标签中 name属性的值FILES中的每一个value同时也是一个标准的python字典对象包含下面三个Keys
#
# filename 上传文件名用字符串表示
# content_type: 上传文件的Content Type
# content 上传文件的原始内容
#
#
# user 是一个django.contrib.auth.models.User对象代表当前登陆的用户。如果访问用户当前
# 没有登陆user将被初始化为django.contrib.auth.models.AnonymousUser的实例。你
# 可以通过user的is_authenticated()方法来辨别用户是否登陆
# if req.user.is_authenticated();只有激活Django中的AuthenticationMiddleware
# 时该属性才可用
#
# session 唯一可读写的属性代表当前会话的字典对象自己有激活Django中的session支持时该属性才可用。#方法
get_full_path(), 比如http://127.0.0.1:8000/index33/?name123 ,req.get_full_path()得到的结果就是/index33/?name123
req.path:/index33 注意一个常用方法request.POST.getlist() 2 HttpResponse对象 对于HttpRequest对象来说是由django自动创建的但是HttpResponse对象就必须我们自己创建。每个view请求处理方法必须返回一个HttpResponse对象。 HttpResponse类在django.http.HttpResponse 在HttpResponse对象上扩展的常用方法 页面渲染 render()推荐br render_to_response(),页面跳转 redirect(路径)locals() 可以直接将函数中所有的变量传给模板 补充 -----------------------------------url.pyurl(rlogin, views.login),url(ryuan_back, views.yuan_back),-----------------------------------views.py
def login(req):if req.methodPOST:if 1:# return redirect(/yuan_back/)nameyuanhaoreturn render(req,my backend.html,locals())return render(req,login.html,locals())def yuan_back(req):name苑昊return render(req,my backend.html,locals())-----------------------------------login.htmlform action/login/ methodpostp姓名input typetext nameusername/pp性别input typetext namesex/pp邮箱input typetext nameemail/ppinput typesubmit valuesubmit/p
/form
-----------------------------------my backend.html
h1用户{{ name }}你好/h1#总结: render和redirect的区别:
# 1 if render的页面需要模板语言渲染,需要的将数据库的数据加载到html,那么所有的这一部分
# 除了写在yuan_back的视图函数中,必须还要写在login中,代码重复,没有解耦.# 2 the most important: url没有跳转到/yuan_back/,而是还在/login/,所以当刷新后
# 又得重新登录. 七 Template基础 模板系统的介绍 你可能已经注意到我们在例子视图中返回文本的方式有点特别。 也就是说HTML被直接硬编码在 Python代码之中。 def current_datetime(request):now datetime.datetime.now()html htmlbodyIt is now %s./body/html % nowreturn HttpResponse(html) 尽管这种技术便于解释视图是如何工作的但直接将HTML硬编码到你的视图里却并不是一个好主意。 让我们来看一下为什么 对页面设计进行的任何改变都必须对 Python 代码进行相应的修改。 站点设计的修改往往比底层 Python 代码的修改要频繁得多因此如果可以在不进行 Python 代码修改的情况下变更设计那将会方便得多。 Python 代码编写和 HTML 设计是两项不同的工作大多数专业的网站开发环境都将他们分配给不同的人员甚至不同部门来完成。 设计者和HTML/CSS的编码人员不应该被要求去编辑Python的代码来完成他们的工作。 程序员编写 Python代码和设计人员制作模板两项工作同时进行的效率是最高的远胜于让一个人等待另一个人完成对某个既包含 Python又包含 HTML 的文件的编辑工作。 基于这些原因将页面的设计和Python的代码分离开会更干净简洁更容易维护。 我们可以使用 Django的 模板系统 (Template System)来实现这种模式这就是本章要具体讨论的问题。 ------------------------------------------------------------------模板语法------------------------------------------------------------------ 一模版的组成 组成HTML代码逻辑控制代码 二 逻辑控制代码的组成 1 变量使用双大括号来引用变量 语法格式 {{var_name}} Template和Context对象 python manange.py shell (进入该django项目的环境)from django.template import Context, Templatet Template(My name is {{ name }}.)c Context({name: Stephane})t.render(c)
My name is Stephane.# 同一模板多个上下文一旦有了模板对象你就可以通过它渲染多个context无论何时我们都可以
# 像这样使用同一模板源渲染多个context只进行 一次模板创建然后多次调用render()方法渲染会
# 更为高效
# Low
for name in (John, Julie, Pat):t Template(Hello, {{ name }})print t.render(Context({name: name}))# Good
t Template(Hello, {{ name }})
for name in (John, Julie, Pat):print t.render(Context({name: name})) Django 模板解析非常快捷。 大部分的解析工作都是在后台通过对简短正则表达式一次性调用来完成。 这和基于 XML 的模板引擎形成鲜明对比那些引擎承担了 XML 解析器的开销且往往比 Django 模板渲染引擎要慢上几个数量级。 from django.shortcuts import render,HttpResponse
from django.template.loader import get_template #记得导入
# Create your views here.import datetime
from django.template import Template,Context# def current_time(req):#原始的视图函数# nowdatetime.datetime.now()# htmlhtmlbody现在时刻h1%s./h1/body/html %now# return HttpResponse(html)# def current_time(req):#django模板修改的视图函数
# nowdatetime.datetime.now()
# tTemplate(htmlbody现在时刻是:h1 stylecolor:red{{current_date}}/h1/body/html)#tget_template(current_datetime.html)
# cContext({current_date:now})
# htmlt.render(c)
# return HttpResponse(html)#另一种写法(推荐)def current_time(req):nowdatetime.datetime.now()return render(req, current_datetime.html, {current_date:now})推荐方式 深度变量的查找万能的句点号 在到目前为止的例子中我们通过 context 传递的简单参数值主要是字符串然而模板系统能够非常简洁地处理更加复杂的数据结构例如list、dictionary和自定义的对象。 在 Django 模板中遍历复杂数据结构的关键是句点字符 (.)。 #最好是用几个例子来说明一下。
# 首先句点可用于访问列表索引例如 from django.template import Template, Contextt Template(Item 2 is {{ items.2 }}.)c Context({items: [apples, bananas, carrots]})t.render(c)
Item 2 is carrots.#假设你要向模板传递一个 Python 字典。 要通过字典键访问该字典的值可使用一个句点from django.template import Template, Contextperson {name: Sally, age: 43}t Template({{ person.name }} is {{ person.age }} years old.)c Context({person: person})t.render(c)
Sally is 43 years old.#同样也可以通过句点来访问对象的属性。 比方说 Python 的 datetime.date 对象有
#year 、 month 和 day 几个属性你同样可以在模板中使用句点来访问这些属性 from django.template import Template, Contextimport datetimed datetime.date(1993, 5, 2)d.yeard.monthd.dayt Template(The month is {{ date.month }} and the year is {{ date.year }}.)c Context({date: d})t.render(c)
The month is 5 and the year is 1993.# 这个例子使用了一个自定义的类演示了通过实例变量加一点(dots)来访问它的属性这个方法适
# 用于任意的对象。from django.template import Template, Contextclass Person(object):
... def __init__(self, first_name, last_name):
... self.first_name, self.last_name first_name, last_namet Template(Hello, {{ person.first_name }} {{ person.last_name }}.)c Context({person: Person(John, Smith)})t.render(c)
Hello, John Smith.# 点语法也可以用来引用对象的方法。 例如每个 Python 字符串都有 upper() 和 isdigit()
# 方法你在模板中可以使用同样的句点语法来调用它们from django.template import Template, Contextt Template({{ var }} -- {{ var.upper }} -- {{ var.isdigit }})t.render(Context({var: hello}))
hello -- HELLO -- Falset.render(Context({var: 123}))
123 -- 123 -- True# 注意这里调用方法时并* 没有* 使用圆括号 而且也无法给该方法传递参数你只能调用不需参数的
# 方法。 变量的过滤器(filter)的使用 语法格式 {{obj|filter:param}} # 1 add 给变量加上相应的值## 2 addslashes : 给变量中的引号前加上斜线## 3 capfirst : 首字母大写## 4 cut 从字符串中移除指定的字符## 5 date 格式化日期字符串## 6 default 如果值是False,就替换成设置的默认值否则就是用本来的值## 7 default_if_none: 如果值是None就替换成设置的默认值否则就使用本来的值#实例:#value1aBcDe
{{ value1|upper }}br#value25
{{ value2|add:3 }}br#value3he llo wo r ld
{{ value3|cut: }}br#import datetime
#value4datetime.datetime.now()
{{ value4|date:Y-m-d }}br#value5[]
{{ value5|default:空的 }}br#value6a href#跳转/a{{ value6 }}{% autoescape off %}{{ value6 }}
{% endautoescape %}{{ value6|safe }}br{{ value6|striptags }}#value71234
{{ value7|filesizeformat }}br
{{ value7|first }}br
{{ value7|length }}br
{{ value7|slice::-1 }}br#value8http://www.baidu.com/?a1b3
{{ value8|urlencode }}brvalue9hello I am yuan 2 标签(tag)的使用使用大括号和百分比的组合来表示使用tag {% tags %} {% if %} 的使用 {% if %}标签计算一个变量值如果是“true”即它存在、不为空并且不是false的boolean值,系统则会显示{% if %}和{% endif %}间的所有内容 {% if num 100 and 8 %}{% if num 200 %}pnum大于200/p{% else %}pnum大于100小于200/p{% endif %}{% elif num 100%}pnum小于100/p{% else %}pnum等于100/p{% endif %}{% if %} 标签接受andor或者not来测试多个变量值或者否定一个给定的变量
{% if %} 标签不允许同一标签里同时出现and和or否则逻辑容易产生歧义例如下面的标签是不合法的{% if obj1 and obj2 or obj3 %} {% for %}的使用 {% for %}标签允许你按顺序遍历一个序列中的各个元素,每次循环模板系统都会渲染{% for %}和{% endfor %}之间的所有内容 ul
{% for obj in list %}li{{ obj.name }}/li
{% endfor %}
/ul#在标签里添加reversed来反序循环列表{% for obj in list reversed %}...{% endfor %}#{% for %}标签可以嵌套{% for country in countries %}h1{{ country.name }}/h1ul{% for city in country.city_list %}li{{ city }}/li{% endfor %}/ul{% endfor %}#系统不支持中断循环系统也不支持continue语句{% for %}标签内置了一个forloop模板变量
#这个变量含有一些属性可以提供给你一些关于循环的信息1forloop.counter表示循环的次数它从1开始计数第一次循环设为1:{% for item in todo_list %}p{{ forloop.counter }}: {{ item }}/p{% endfor %}
2forloop.counter0 类似于forloop.counter但它是从0开始计数第一次循环设为0
3forloop.revcounter
4forloop.revcounter0
5forloop.first当第一次循环时值为True在特别情况下很有用{% for object in objects %} {% if forloop.first %}li classfirst{% else %}li{% endif %} {{ object }} /li {% endfor %} # 富有魔力的forloop变量只能在循环中得到当模板解析器到达{% endfor %}时forloop就消失了
# 如果你的模板context已经包含一个叫forloop的变量Django会用{% for %}标签替代它
# Django会在for标签的块中覆盖你定义的forloop变量的值
# 在其他非循环的地方你的forloop变量仍然可用#{% empty %}{{li }}{% for i in li %}li{{ forloop.counter0 }}----{{ i }}/li{% empty %}lithis is empty!/li{% endfor %}# [11, 22, 33, 44, 55]
# 0----11
# 1----22
# 2----33
# 3----44
# 4----55 {%csrf_token%}csrf_token标签 用于生成csrf_token的标签用于防治跨站攻击验证。注意如果你在view的index里用的是render_to_response方法不会生效 其实这里是会生成一个input标签和其他表单标签一起提交给后台的。 {% url %}: 引用路由配置的地址 form action{% url bieming%} input typetextinput typesubmitvalue提交{%csrf_token%}
/form {% with %}:用更简单的变量名替代复杂的变量名 {% with totalfhjsaldfhjsdfhlasdfhljsdal %} {{ total }} {% endwith %} {% verbatim %}: 禁止render {% verbatim %}{{ hello }}{% endverbatim %} {% load %}: 加载标签库 3 自定义filter和simple_tag a、在app中创建templatetags模块(必须的) b、创建任意 .py 文件如my_tags.py from django import template
from django.utils.safestring import mark_saferegister template.Library() #register的名字是固定的,不可改变register.filter
def filter_multi(v1,v2):return v1 * v2register.simple_tag
def simple_tag_multi(v1,v2):return v1 * v2register.simple_tag
def my_input(id,arg):result input typetext id%s class%s / %(id,arg,)return mark_safe(result) c、在使用自定义simple_tag和filter的html文件中导入之前创建的 my_tags.py {% load my_tags %} d、使用simple_tag和filter如何调用 -------------------------------.html
{% load xxx %} #首行# num12
{{ num|filter_multi:2 }} #24{{ num|filter_multi:[22,333,4444] }}{% simple_tag_multi 2 5 %} 参数不限,但不能放在if for语句中
{% simple_tag_multi num 5 %} e、在settings中的INSTALLED_APPS配置当前app不然django无法找到自定义的simple_tag. 注意 filter可以用在if等语句后simple_tag不可以 {% if num|filter_multi:30 100 %}{{ num|filter_multi:30 }}
{% endif %} 4 extend模板继承 include 模板标签 在讲解了模板加载机制之后我们再介绍一个利用该机制的内建模板标签 {% include %} 。该标签允许在模板中包含其它的模板的内容。 标签的参数是所要包含的模板名称可以是一个变量也可以是用单/双引号硬编码的字符串。 每当在多个模板中出现相同的代码时就应该考虑是否要使用 {% include %} 来减少重复。 extend(继承)模板标签 到目前为止我们的模板范例都只是些零星的 HTML 片段但在实际应用中你将用 Django 模板系统来创建整个 HTML 页面。 这就带来一个常见的 Web 开发问题 在整个网站中如何减少共用页面区域比如站点导航所引起的重复和冗余代码 解决该问题的传统做法是使用 服务器端的 includes 你可以在 HTML 页面中使用该指令将一个网页嵌入到另一个中。 事实上 Django 通过刚才讲述的 {% include %} 支持了这种方法。 但是用 Django 解决此类问题的首选方法是使用更加优雅的策略—— 模板继承 。 本质上来说模板继承就是先构造一个基础框架模板而后在其子模板中对它所包含站点公用部分和定义块进行重载。 让我们通过修改 current_datetime.html 文件为 current_datetime 创建一个更加完整的模板来体会一下这种做法 !DOCTYPE HTML PUBLIC -//W3C//DTD HTML 4.01//EN
html langen
headtitleThe current time/title
/head
bodyh1My helpful timestamp site/h1pIt is now {{ current_date }}./phrpThanks for visiting my site./p
/body
/html 这看起来很棒但如果我们要为 hours_ahead 视图创建另一个模板会发生什么事情呢 !DOCTYPE HTML PUBLIC -//W3C//DTD HTML 4.01//EN
html langen
headtitleFuture time/title
/head
bodyh1My helpful timestamp site/h1pIn {{ hour_offset }} hour(s), it will be {{ next_time }}./phrpThanks for visiting my site./p
/body
/html 很明显我们刚才重复了大量的 HTML 代码。 想象一下如果有一个更典型的网站它有导航条、样式表可能还有一些 JavaScript 代码事情必将以向每个模板填充各种冗余的 HTML 而告终。 解决这个问题的服务器端 include 方案是找出两个模板中的共同部分将其保存为不同的模板片段然后在每个模板中进行 include。 也许你会把模板头部的一些代码保存为 header.html 文件 !DOCTYPE HTML PUBLIC -//W3C//DTD HTML 4.01//EN
html langen
head 你可能会把底部保存到文件 footer.html : hrpThanks for visiting my site./p
/body
/html 对基于 include 的策略头部和底部的包含很简单。 麻烦的是中间部分。 在此范例中每个页面都有一个h1My helpful timestamp site/h1 标题但是这个标题不能放在 header.html 中因为每个页面的 title 是不同的。 如果我们将 h1 包含在头部我们就不得不包含 title 但这样又不允许在每个页面对它进行定制。 何去何从呢 Django 的模板继承系统解决了这些问题。 你可以将其视为服务器端 include 的逆向思维版本。 你可以对那些不同 的代码段进行定义而不是 共同 代码段。 第一步是定义 基础模板,该框架之后将由子模板所继承。 以下是我们目前所讲述范例的基础模板 !DOCTYPE HTML PUBLIC -//W3C//DTD HTML 4.01//EN
html langen
headtitle{% block title %}{% endblock %}/title
/head
bodyh1My helpful timestamp site/h1{% block content %}{% endblock %}{% block footer %}hrpThanks for visiting my site./p{% endblock %}
/body
/html 这个叫做 base.html 的模板定义了一个简单的 HTML 框架文档我们将在本站点的所有页面中使用。 子模板的作用就是重载、添加或保留那些块的内容。 如果你一直按顺序学习到这里保存这个文件到你的template目录下命名为 base.html . 我们使用模板标签 {% block %} 。 所有的 {% block %} 标签告诉模板引擎子模板可以重载这些部分。 每个{% block %}标签所要做的是告诉模板引擎该模板下的这一块内容将有可能被子模板覆盖。 现在我们已经有了一个基本模板我们可以修改 current_datetime.html 模板来 使用它 {% extends base.html %}{% block title %}The current time{% endblock %}{% block content %}
pIt is now {{ current_date }}./p
{% endblock %} 再为 hours_ahead 视图创建一个模板看起来是这样的 {% extends base.html %}{% block title %}Future time{% endblock %}{% block content %}
pIn {{ hour_offset }} hour(s), it will be {{ next_time }}./p
{% endblock %} 看起来很漂亮是不是 每个模板只包含对自己而言 独一无二 的代码。 无需多余的部分。 如果想进行站点级的设计修改仅需修改 base.html 所有其它模板会立即反映出所作修改。 以下是其工作方式 在加载 current_datetime.html 模板时模板引擎发现了 {% extends %} 标签 注意到该模板是一个子模板。 模板引擎立即装载其父模板即本例中的 base.html 。此时模板引擎注意到 base.html 中的三个 {% block %} 标签并用子模板的内容替换这些 block 。因此引擎将会使用我们在 { block title %} 中定义的标题对 {% block content %} 也是如此。 所以网页标题一块将由{% block title %}替换同样地网页的内容一块将由 {% block content %}替换。 注意由于子模板并没有定义 footer 块模板系统将使用在父模板中定义的值。 父模板 {% block %} 标签中的内容总是被当作一条退路。继承并不会影响到模板的上下文。 换句话说任何处在继承树上的模板都可以访问到你传到模板中的每一个模板变量。你可以根据需要使用任意多的继承次数。 使用继承的一种常见方式是下面的三层法 1 创建 base.html 模板在其中定义站点的主要外观感受。 这些都是不常修改甚至从不修改的部分。2 为网站的每个区域创建 base_SECTION.html 模板(例如, base_photos.html 和 base_forum.html )。这些模板对base.html 进行拓展并包含区域特定的风格与设计。3 为每种类型的页面创建独立的模板例如论坛页面或者图片库。 这些模板拓展相应的区域模板。 这个方法可最大限度地重用代码并使得向公共区域如区域级的导航添加内容成为一件轻松的工作。 以下是使用模板继承的一些诀窍 1如果在模板中使用 {% extends %} 必须保证其为模板中的第一个模板标记。 否则模板继承将不起作用。2一般来说基础模板中的 {% block %} 标签越多越好。 记住子模板不必定义父模板中所有的代码块因此你可以用合理的缺省值对一些代码块进行填充然后只对子模板所需的代码块进行重定义。 俗话说钩子越多越好。3如果发觉自己在多个模板之间拷贝代码你应该考虑将该代码段放置到父模板的某个 {% block %} 中。如果你需要访问父模板中的块的内容使用 {{ block.super }}这个标签吧这一个魔法变量将会表现出父模板中的内容。 如果只想在上级代码块基础上添加内容而不是全部重载该变量就显得非常有用了。4不允许在同一个模板中定义多个同名的 {% block %} 。 存在这样的限制是因为block 标签的工作方式是双向的。也就是说block 标签不仅挖了一个要填的坑也定义了在父模板中这个坑所填充的内容。如果模板中出现了两个相同名称的 {% block %} 标签父模板将无从得知要使用哪个块的内容。 转载于:https://www.cnblogs.com/huangjinshan/p/6479929.html