第1章 Django之MTV模型
1.1 课堂总结记录
Django1流程:
(1) 设计url: url(r'^timer/', timer), #timer(request)
(2) 构建视图函数:
在views.py:deftimer(request):importtime
ctime=time.time()return render(request,"timer.html",{"ctime":ctime})
(3) templates:timer.html
当前时间:{{ ctime }}
2URL控制器(路由层):
简单配置
分组
有名分组
分发
反向解析3 MTV--View(视图层):
请求对象
request:
request.GET
request.POST
request.method
request.path
响应对象:
Httpresponse()
render()
redirect()4 MTV--Template(模板层)
渲染变量 {{ }}
深度查询句点符
过滤器{{ var|filter:参数}}
add
date
safe
filesizeformat
渲染标签 {% %}for
ifurl 反向解析
csrftoken
自定义过滤器和标签4 MTV--Model(模型层)
ORM---对象关系映射classBook(model.Model):
title=models.CharFiled(max_length=32)
create table book (
title varchar(32)
)
类名------表名
类属性------表字段
类实例对象------表记录
单表记录操作:
Book
id title1python2java
添加记录:#方式1:
#obj新添加记录对象
obj=Book.objects.create(title="python")#方式2:
obj=Book(title="java")
obj.save()
查询记录API(QuerySet):<1>all(): 查询所有结果<2> filter(**kwargs): 它包含了与所给筛选条件相匹配的对象<3> get(**kwargs): 返回与所给筛选条件相匹配的对象,返回结果有且只有一个,
如果符合筛选条件的对象超过一个或者没有都会抛出错误。<5> exclude(**kwargs): 它包含了与所给筛选条件不匹配的对象<4> values(*field): 返回一个ValueQuerySet——一个特殊的QuerySet,运行后得到的并不是一系列
model的实例化对象,而是一个可迭代的字典序列<9> values_list(*field): 它与values()非常相似,它返回的是一个元组序列,values返回的是一个字典序列<6> order_by(*field): 对查询结果排序<7>reverse(): 对查询结果反向排序<8>distinct(): 从返回结果中剔除重复纪录<10>count(): 返回数据库中匹配查询(QuerySet)的对象数量。<11>first(): 返回第一条记录<12>last(): 返回最后一条记录<13>exists(): 如果QuerySet包含数据,就返回True,否则返回False
模糊查询: __
删除记录
Book.objects.filter(price=123,title="python").delete()
更新记录:
Book.objects.filter(price=123,title="python").update(title="python123")
mysql表关系
一对多关系:关联字段
多对多关系:创建关系表
Book
title=...
publish=Foreignkey(to="Publish",to_field="")
authors=Publish
name=....
Author
name=....
Book2Author
课堂笔记
1.2 MTV模型的含义
Django的MTV分别代表:
Model(模型):和数据库相关的,负责业务对象与数据库的对象(ORM)
Template(模板):放所有的html文件,负责如何把页面展示给用户
模板语法:目的是将白变量(数据库的内容)如何巧妙的嵌入到html页面中
View(视图):负责业务逻辑,并在适当的时候调用Model和Template
此外,Django还有一个URL分发器。它的作用是将一个个URL的页面请求分别发给不同的Views处理,Views再调用相应的Model和Template。
1.3 Django基本命令
1.3.1 下载Django:
pip3 install django
1.4 创建一个django project
创建一个Django项目分两种方式创建:
1.4.1 第一种方法使用pycharm软件创建
文件——新建项目——切到Django
创建成功后会自动生成这样一个工程。目录结构如下:
1.5 第二种命令行创建
1.5.1 创建一个Django对象
django-admin.py startproject 项目名称
django-admin startproject mysite
创建成功后会生成这样一个工程。目录结构如下:
manage.py------启动文件 (Django项目里面的工具,通过它可以调用Django shell的数目和数据库等)
settings.py------包含了项目的一些设置,包括数据库信息、调试标志以及其他一些工作的变量。
urls.py-----------路径与视图函数的映射关系
1.5.2 在mysite目录下创建应用
python3 manage.py startapp blog(应用名称)
创建成功后会生成这样一个工程。目录结构如下:
1.5.3 启动django项目
python3 manage.py runserver 8080
这样我们的django就启动起来了!当我们访问:http://127.0.0.1:8080/时就可以看到:
1.6 创建表命令
1.6.1 同步更改数据库表或字段
'''python3 manage.py syncdb
注意:Django 1.7.1 及以上的版本需要用以下命令
python3 manage.py makemigrations
python3 manage.py migrate'''
这种方法可以创建表,当你在models.py中新增了类时,运行它就可以自动在数据库中创建表了,不用手动创建。
1.6.2 清空数据库
python3 manage.py flush
此命令会询问是 yes 还是 no, 选择 yes 会把数据全部清空掉,只留下空表。
1.6.3 创建超级管理员
'''python3 manage.py createsuperuser
# 按照提示输入用户名和对应的密码就好了邮箱可以留空,用户名和密码必填
# 修改 用户密码可以用:
python3 manage.py changepassword username'''
1.6.4 Django 项目环境终端
python3 manage.py shell
这个命令和 直接运行 python 进入 shell 的区别是:你可以在这个 shell 里面调用当前项目的 models.py 中的 API,对于操作数据的测试非常方便。
1.6.5 Django 项目环境终端
python3 manage.py dbshell
Django 会自动进入在settings.py中设置的数据库,如果是 MySQL 或 postgreSQL,会要求输入数据库用户密码。
在这个终端可以执行数据库的SQL语句。如果您对SQL比较熟悉,可能喜欢这种方式。
1.6.6 更多命令
python3 manage.py #########切换到项目目录下
查看所有的命令,忘记子名称的时候特别有用。
1.6.7 静态文件配置
概述:
静态文件交由Web服务器处理,Django本身不处理静态文件。简单的处理逻辑如下(以nginx为例):
URI请求----->按照Web服务器里面的配置规则先处理,以nginx为例,主要求配置在nginx.
conf里的location|---------->如果是静态文件,则由nginx直接处理|---------->如果不是则交由Django处理,Django根据urls.py里面的规则进行匹配
以上是部署到Web服务器后的处理方式,为了便于开发,Django提供了在开发环境的对静态文件的处理机制,方法是这样:
概述
static配置:
STATIC主要指的是如css,js,images这样文件:
STATIC_URL = '/static/' #别名
STATICFILES_DIRS=(
os.path.join(BASE_DIR,"static"), #实际名 ,即实际文件夹的名字
)'''注意点1:
django对引用名和实际名进行映射,引用时,只能按照引用名来,不能按实际名去找
------error-----不能直接用,必须用STATIC_URL = '/static/':
注意点2:
STATICFILES_DIRS = (
("app01",os.path.join(BASE_DIR, "app01/statics")),
)
'''
View Code
have a try:
http://127.0.0.1:8000/static/jquery.js
media配置:
#in settings:
MEDIA_URL="/media/"MEDIA_ROOT=os.path.join(BASE_DIR,"app01","media","upload")#in urls:
from django.views.static importserve
url(r'^media/(?P.*)$', serve, {'document_root': settings.MEDIA_ROOT}),
View Code
have a try:
http://127.0.0.1:8000/media/1.png
'''静态文件的处理又包括STATIC和MEDIA两类,这往往容易混淆,在Django里面是这样定义的:
MEDIA:指用户上传的文件,比如在Model里面的FileFIeld,ImageField上传的文件。如果你定义
MEDIA_ROOT=c:\temp\media,那么File=models.FileField(upload_to="abc/")#,上传的文件就会被保存到c:\temp\media\abc
eg:
class blog(models.Model):
Title=models.charField(max_length=64)
Photo=models.ImageField(upload_to="photo")
上传的图片就上传到c:\temp\media\photo,而在模板中要显示该文件,则在这样写
在settings里面设置的MEDIA_ROOT必须是本地路径的绝对路径,一般是这样写:
BASE_DIR= os.path.abspath(os.path.dirname(__file__))
MEDIA_ROOT=os.path.join(BASE_DIR,'media/').replace('\\','/')
MEDIA_URL是指从浏览器访问时的地址前缀,举个例子:
MEDIA_ROOT=c:\temp\media\photo
MEDIA_URL="/data/"
在开发阶段,media的处理由django处理:
访问http://localhost/data/abc/a.png就是访问c:\temp\media\photo\abc\a.png
在模板里面这样写
在部署阶段最大的不同在于你必须让web服务器来处理media文件,因此你必须在web服务器中配置,
以便能让web服务器能访问media文件
以nginx为例,可以在nginx.conf里面这样:
location ~/media/{
root/temp/
break;
}
具体可以参考如何在nginx部署django的资料。'''
View Code
第2章 视图层之路由配置系统(views)
URL配置(URLconf)就像Django 所支撑网站的目录。它的本质是URL与要为该URL调用的视图函数之间的映射表;你就是以这种方式告诉Django,对于这个URL调用这段代码,对于那个URL调用那段代码。
'''urlpatterns = [
url(正则表达式, views视图函数,参数,别名),
]
参数说明:
一个正则表达式字符串
一个可调用对象,通常为一个视图函数或一个指定视图函数路径的字符串
可选的要传递给视图函数的默认参数(字典形式)
一个可选的name参数'''
2.1 URLconf的正则字符串参数
2.1.1 简单配置
######项目cms中的urls.py中的内容
from django.conf.urls importurlfrom . importviews
urlpatterns=[
url(r'^articles/2003/$', views.special_case_2003),
url(r'^articles/([0-9]{4})/$', views.year_archive),
url(r'^articles/([0-9]{4})/([0-9]{2})/$', views.month_archive),
url(r'^articles/([0-9]{4})/([0-9]{2})/([0-9]+)/$', views.article_detail),
]
项目cms中的urls.py中的内容
#############说明
'''NOTE:
1 一旦匹配成功则不再继续
2 若要从URL 中捕获一个值,只需要在它周围放置一对圆括号。
3 不需要添加一个前导的反斜杠,因为每个URL 都有。例如,应该是^articles 而不是 ^/articles。
4 每个正则表达式前面的'r' 是可选的但是建议加上。
一些请求的例子:
/articles/2005/3/ 不匹配任何URL 模式,因为列表中的第三个模式要求月份应该是两个数字。
/articles/2003/ 将匹配列表中的第一个模式不是第二个,因为模式按顺序匹配,第一个会首先测试是否匹配。
/articles/2005/03/ 请求将匹配列表中的第三个模式。Django 将调用函数
views.month_archive(request, '2005', '03')。'''
说明
#设置项是否开启URL访问地址后面不为/跳转至带有/的路径
APPEND_SLASH=True
2.1.2 无名分组和有名分组(named group)
上面的示例使用简单的、没有命名的正则表达式组(通过圆括号)来捕获URL 中的值并以位置 参数传递给视图。在更高级的用法中,可以使用命名的正则表达式组来捕获URL 中的值并以关键字 参数传递给视图。
在Python 正则表达式中,命名正则表达式组的语法是(?Ppattern),其中name 是组的名称,pattern 是要匹配的模式。
下面是以上URLconf 使用命名组的重写:
from app02 importviewsfrom django.conf.urls importurl
urlpatterns=[#无名分组
url(r'^article/\d{4}', views.year) ,
url(r'^article/(\d{4})$', views.year2),
如果有多个匹配一样的时候,谁放在上面就匹配谁,上面的就会把下面的覆盖了
正则加上括号,就是分组,会把分组的内容作为year2函数的参数传进去
url(r'^article/(\d{4})/(\d{2})$', views.year_month),#有名分组(就是给分组起个名字,这样定义的好处就是按照关键字参数去传参了,指名道姓的方式)
url(r'^article/(?P\d{4})/(?P\d{2})$', views.year_month_hasname)
]
View Code
这个实现与前面的示例完全相同,只有一个细微的差别:捕获的值作为关键字参数而不是位置参数传递给视图函数。例如:
/articles/2005/03/请求将调用views.month_archive(request, year='2005', month='03')函数/articles/2003/03/03/请求将调用函数views.article_detail(request, year='2003', month='03', day='03')。
在实际应用中,这意味你的URLconf 会更加明晰且不容易产生参数顺序问题的错误 —— 你可以在你的视图函数定义中重新安排参数的顺序。当然,这些好处是以简洁为代价;有些开发人员认为命名组语法丑陋而繁琐。
2.1.3 URLconf 在什么上查找
URLconf 在请求的URL 上查找,将它当做一个普通的Python 字符串。不包括GET和POST参数以及域名。
例如,http://www.example.com/myapp/ 请求中,URLconf 将查找myapp/。
在http://www.example.com/myapp/?page=3 请求中,URLconf 仍将查找myapp/。
URLconf 不检查请求的方法。换句话讲,所有的请求方法 —— 同一个URL的POST、GET、HEAD等等 —— 都将路由到相同的函数。
2.1.4 捕获的参数永远是字符串
每个捕获的参数都作为一个普通的Python 字符串传递给视图,无论正则表达式使用的是什么匹配方式。例如,下面这行URLconf 中:
url(r'^articles/(?P[0-9]{4})/$', views.year_archive),
views.year_archive() 的year 参数将是一个字符串
2.1.5 指定视图参数的默认值
有一个方便的小技巧是指定视图参数的默认值。 下面是一个URLconf 和视图的示例:
#URLconf
from django.conf.urls importurlfrom . importviews
urlpatterns=[
url(r'^blog/$', views.page),
url(r'^blog/page(?P[0-9]+)/$', views.page),
]#View (in blog/views.py)
def page(request, num="1"):
...
View Code
在上面的例子中,两个URL模式指向同一个视图views.page —— 但是第一个模式不会从URL 中捕获任何值。如果第一个模式匹配,page() 函数将使用num参数的默认值"1"。如果第二个模式匹配,page() 将使用正则表达式捕获的num 值。
2.1.6 Including other URLconfs
#At any point, your urlpatterns can “include” other URLconf modules. This
#essentially “roots” a set of URLs below other ones.
#For example, here’s an excerpt of the URLconf for the Django website itself.
#It includes a number of other URLconfs:
from django.conf.urls importinclude, url
urlpatterns=[
url(r'^admin/', admin.site.urls),
url(r'^blog/', include('blog.urls')),
]
View Code
2.2 传递额外的选项给视图函数(了解)
URLconfs 具有一个钩子,让你传递一个Python 字典作为额外的参数传递给视图函数。
django.conf.urls.url() 函数可以接收一个可选的第三个参数,它是一个字典,表示想要传递给视图函数的额外关键字参数。
例如:
from django.conf.urls importurlfrom . importviews
urlpatterns=[
url(r'^blog/(?P[0-9]{4})/$', views.year_archive, {'foo': 'bar'}),
]
在这个例子中,对于/blog/2005/请求,Django 将调用views.year_archive(request, year='2005', foo='bar')。
这个技术在Syndication 框架中使用,来传递元数据和选项给视图。
2.3 URL 的反向解析
在使用Django 项目时,一个常见的需求是获得URL 的最终形式,以用于嵌入到生成的内容中(视图中和显示给用户的URL等)或者用于处理服务器端的导航(重定向等)。
人们强烈希望不要硬编码这些URL(费力、不可扩展且容易产生错误)或者设计一种与URLconf 毫不相关的专门的URL 生成机制,因为这样容易导致一定程度上产生过期的URL。
换句话讲,需要的是一个DRY 机制。除了其它有点,它还允许设计的URL 可以自动更新而不用遍历项目的源代码来搜索并替换过期的URL。
获取一个URL 最开始想到的信息是处理它视图的标识(例如名字),查找正确的URL 的其它必要的信息有视图参数的类型(位置参数、关键字参数)和值。
Django 提供一个办法是让URL 映射是URL 设计唯一的地方。你填充你的URLconf,然后可以双向使用它:
l 根据用户/浏览器发起的URL 请求,它调用正确的Django 视图,并从URL 中提取它的参数需要的值。
l 根据Django 视图的标识和将要传递给它的参数的值,获取与之关联的URL。
第一种方式是我们在前面的章节中一直讨论的用法。第二种方式叫做反向解析URL、反向URL 匹配、反向URL 查询或者简单的URL 反查。
在需要URL 的地方,对于不同层级,Django 提供不同的工具用于URL 反查:
l 在模板中:使用url 模板标签。
l 在Python 代码中:使用django.core.urlresolvers.reverse() 函数。
l 在更高层的与处理Django 模型实例相关的代码中:使用get_absolute_url() 方法。
例子:
考虑下面的URLconf:
from django.conf.urls importurlfrom . importviews
urlpatterns=[#...
url(r'^articles/([0-9]{4})/$', views.year_archive, name='news-year-archive'),#...
]
View Code
根据这里的设计,某一年nnnn对应的归档的URL是/articles/nnnn/。
你可以在模板的代码中使用下面的方法获得它们:
- {% for yearvar in year_list %}
- {{ yearvar }} Archive{% endfor %}
在Python 代码中,这样使用:
from django.core.urlresolvers importreversefrom django.http importHttpResponseRedirectdefredirect_to_year(request):#...
year= 2006
#...
return HttpResponseRedirect(reverse('news-year-archive', args=(year,)))
View Code
如果出于某种原因决定按年归档文章发布的URL应该调整一下,那么你将只需要修改URLconf 中的内容。
在某些场景中,一个视图是通用的,所以在URL 和视图之间存在多对一的关系。对于这些情况,当反查URL 时,只有视图的名字还不够。
2.4 名称空间(Namespace)
命名空间(英语:Namespace)是表示标识符的可见范围。一个标识符可在多个命名空间中定义,它在不同命名空间中的含义是互不相干的。这样,在一个新的命名空间中可定义任何标识符,它们不会与任何已有的标识符发生冲突,因为已有的定义都处于其它命名空间中。
由于name没有作用域,Django在反解URL时,会在项目全局顺序搜索,当查找到第一个name指定URL时,立即返回
我们在开发项目时,会经常使用name属性反解出URL,当不小心在不同的app的urls中定义相同的name时,可能会导致URL反解错误,为了避免这种事情发生,引入了命名空间。
project.urls:
urlpatterns =[
url(r'^admin/', admin.site.urls),
url(r'^app01/', include("app01.urls",namespace="app01")),
url(r'^app02/', include("app02.urls",namespace="app02")),
]
View Code
app01.urls:
urlpatterns =[
url(r'^index/', index,name="index"),
]
app02.urls:
urlpatterns =[
url(r'^index/', index,name="index"),
]
app01.views
from django.core.urlresolvers importreversedefindex(request):return HttpResponse(reverse("app01:index"))
app02.viewsfrom django.core.urlresolvers importreversedefindex(request):return HttpResponse(reverse("app02:index"))
View Code
2.5 CBV
url(r'^login.html$', views.Login.as_view()),
============================from django.views importViewclassLogin(View):def dispatch(self, request, *args, **kwargs):print('before')
obj= super(Login,self).dispatch(request, *args, **kwargs)print('after')returnobjdefget(self,request):return render(request,'login.html')defpost(self,request):print(request.POST.get('user'))return HttpResponse('Login.post')
View Code
第3章 视图层之视图函数(views)
一个视图函数,简称视图,是一个简单的Python 函数,它接受Web请求并且返回Web响应。响应可以是一张网页的HTML内容,一个重定向,一个404错误,一个XML文档,或者一张图片. . . 是任何东西都可以。无论视图本身包含什么逻辑,都要返回响应。代码写在哪里也无所谓,只要它在你的Python目录下面。除此之外没有更多的要求了——可以说“没有什么神奇的地方”。为了将代码放在某处,约定是将视图放置在项目或应用程序目录中的名为views.py的文件中。
3.1 一个简单的视图
3.1.1 实例一
下面是一个返回当前日期和时间作为HTML文档的视图:
from django.http importHttpResponseimportdatetimedefcurrent_datetime(request):
now=datetime.datetime.now()
html= "
It is now %s." %nowreturnHttpResponse(html)View Code
让我们逐行阅读上面的代码:
l 首先,我们从 django.http模块导入了HttpResponse类,以及Python的datetime库。
l 接着,我们定义了current_datetime函数。它就是视图函数。每个视图函数都使用HttpRequest对象作为第一个参数,并且通常称之为request。
l 注意,视图函数的名称并不重要;不需要用一个统一的命名方式来命名,以便让Django识别它。我们将其命名为current_datetime,是因为这个名称能够精确地反映出它的功能。
l 这个视图会返回一个HttpResponse对象,其中包含生成的响应。每个视图函数都负责返回一个HttpResponse对象。
3.1.2 实例二
######################views.py###############################
from django.shortcuts importrender,HttpResponse#Create your views here.
def year(request): #request参数请求所有的参数,这个参数一定要有
return HttpResponse("ok") #每一个视图函数必须有return
defyear2(request,year):print(year)return HttpResponse("hello") #返回的一定是一个字符串,如果你想return纯字符串,就用HttpResponse方法
defyear_month(request,year,month):print(year,month)#返回的是匹配的年和月拼接的结果
return HttpResponse(year+month) #
defyear_month_hasname(request,month,year):#return HttpResponse("ok")
print(year,month)return HttpResponse("month是:%s,year是:%s"%(month,year))
views.py
注意:视图会返回一个HttpResponse对象,其中包含生成的响应。每个视图函数都负责返回一个HttpResponse对象。
3.2 HttpRequest对象的属性和方法
3.2.1 HttpRequest对象的属性
属性:
django将请求报文中的请求行、首部信息、内容主体封装成 HttpRequest 类中的属性。
除了特殊说明的之外,其他均为只读的。'''1.HttpRequest.body
一个字符串,代表请求报文的主体。在处理非 HTTP 形式的报文时非常有用,例如:二进制图片、XML,Json等。
但是,如果要处理表单数据的时候,推荐还是使用 HttpRequest.POST 。
另外,我们还可以用 python 的类文件方法去操作它,详情参考 HttpRequest.read() 。
2.HttpRequest.path
一个字符串,表示请求的路径组件(不含域名)。
例如:"/music/bands/the_beatles/"
3.HttpRequest.method
一个字符串,表示请求使用的HTTP 方法。必须使用大写。
例如:"GET"、"POST"
4.HttpRequest.encoding
一个字符串,表示提交的数据的编码方式(如果为 None 则表示使用 DEFAULT_CHARSET 的设置,默认为 'utf-8')。
这个属性是可写的,你可以修改它来修改访问表单数据使用的编码。
接下来对属性的任何访问(例如从 GET 或 POST 中读取数据)将使用新的 encoding 值。
如果你知道表单数据的编码不是 DEFAULT_CHARSET ,则使用它。
5.HttpRequest.GET
一个类似于字典的对象,包含 HTTP GET 的所有参数。详情请参考 QueryDict 对象。
6.HttpRequest.POST
一个类似于字典的对象,如果请求中包含表单数据,则将这些数据封装成 QueryDict 对象。
POST 请求可以带有空的 POST 字典 —— 如果通过 HTTP POST 方法发送一个表单,但是表单中没有任何的数据,QueryDict 对象依然会被创建。
因此,不应该使用 if request.POST 来检查使用的是否是POST 方法;应该使用 if request.method == "POST"
另外:如果使用 POST 上传文件的话,文件信息将包含在 FILES 属性中。
7.HttpRequest.REQUEST
一个类似于字典的对象,它首先搜索POST,然后搜索GET,主要是为了方便。灵感来自于PHP 的 $_REQUEST。
例如,如果 GET = {"name": "john"} 而 POST = {"age": '34'} , REQUEST["name"] 将等于"john", REQUEST["age"] 将等于"34"。
强烈建议使用 GET 和 POST 而不要用REQUEST,因为它们更加明确。
8.HttpRequest.COOKIES
一个标准的Python 字典,包含所有的cookie。键和值都为字符串。
9.HttpRequest.FILES
一个类似于字典的对象,包含所有的上传文件信息。
FILES 中的每个键为 中的name,值则为对应的数据。
注意,FILES 只有在请求的方法为POST 且提交的
带有enctype="multipart/form-data" 的情况下才会包含数据。否则,FILES 将为一个空的类似于字典的对象。
10.HttpRequest.META
一个标准的Python 字典,包含所有的HTTP 首部。具体的头部信息取决于客户端和服务器,下面是一些示例:
CONTENT_LENGTH —— 请求的正文的长度(是一个字符串)。
CONTENT_TYPE —— 请求的正文的MIME 类型。
HTTP_ACCEPT —— 响应可接收的Content-Type。
HTTP_ACCEPT_ENCODING —— 响应可接收的编码。
HTTP_ACCEPT_LANGUAGE —— 响应可接收的语言。
HTTP_HOST —— 客服端发送的HTTP Host 头部。
HTTP_REFERER —— Referring 页面。
HTTP_USER_AGENT —— 客户端的user-agent 字符串。
QUERY_STRING —— 单个字符串形式的查询字符串(未解析过的形式)。
REMOTE_ADDR —— 客户端的IP 地址。
REMOTE_HOST —— 客户端的主机名。
REMOTE_USER —— 服务器认证后的用户。
REQUEST_METHOD —— 一个字符串,例如"GET" 或"POST"。
SERVER_NAME —— 服务器的主机名。
SERVER_PORT —— 服务器的端口(是一个字符串)。
从上面可以看到,除 CONTENT_LENGTH 和 CONTENT_TYPE 之外,请求中的任何 HTTP 首部转换为 META 的键时,
都会将所有字母大写并将连接符替换为下划线最后加上 HTTP_ 前缀。
所以,一个叫做 X-Bender 的头部将转换成 META 中的 HTTP_X_BENDER 键。
11.HttpRequest.user
一个 AUTH_USER_MODEL 类型的对象,表示当前登录的用户。
如果用户当前没有登录,user 将设置为 django.contrib.auth.models.AnonymousUser 的一个实例。你可以通过 is_authenticated() 区分它们。
例如:
if request.user.is_authenticated():
# Do something for logged-in users.
else:
# Do something for anonymous users.
user 只有当Django 启用 AuthenticationMiddleware 中间件时才可用。
-------------------------------------------------------------------------------------
匿名用户
class models.AnonymousUser
django.contrib.auth.models.AnonymousUser 类实现了django.contrib.auth.models.User 接口,但具有下面几个不同点:
id 永远为None。
username 永远为空字符串。
get_username() 永远返回空字符串。
is_staff 和 is_superuser 永远为False。
is_active 永远为 False。
groups 和 user_permissions 永远为空。
is_anonymous() 返回True 而不是False。
is_authenticated() 返回False 而不是True。
set_password()、check_password()、save() 和delete() 引发 NotImplementedError。
New in Django 1.8:
新增 AnonymousUser.get_username() 以更好地模拟 django.contrib.auth.models.User。
12.HttpRequest.session
一个既可读又可写的类似于字典的对象,表示当前的会话。只有当Django 启用会话的支持时才可用。
完整的细节参见会话的文档。
13.HttpRequest.resolver_match
一个 ResolverMatch 的实例,表示解析后的URL。这个属性只有在 URL 解析方法之后才设置,这意味着它在所有的视图中可以访问,
但是在 URL 解析发生之前执行的中间件方法中不可以访问(比如process_request,但你可以使用 process_view 代替)。'''
View Code
3.2.2 HttpRequest对象的方法
'''1.HttpRequest.get_host()
根据从HTTP_X_FORWARDED_HOST(如果打开 USE_X_FORWARDED_HOST,默认为False)和 HTTP_HOST 头部信息返回请求的原始主机。 如果这两个头部没有提供相应的值,则使用SERVER_NAME 和SERVER_PORT,在PEP 3333 中有详细描述。
USE_X_FORWARDED_HOST:一个布尔值,用于指定是否优先使用 X-Forwarded-Host 首部,仅在代理设置了该首部的情况下,才可以被使用。
例如:"127.0.0.1:8000"
注意:当主机位于多个代理后面时,get_host() 方法将会失败。除非使用中间件重写代理的首部。
2.HttpRequest.get_full_path()
返回 path,如果可以将加上查询字符串。
例如:"/music/bands/the_beatles/?print=true"
3.HttpRequest.build_absolute_uri(location)
返回location 的绝对URI。如果location 没有提供,则使用request.get_full_path()的返回值。
如果URI 已经是一个绝对的URI,将不会修改。否则,使用请求中的服务器相关的变量构建绝对URI。
例如:"http://example.com/music/bands/the_beatles/?print=true"
4.HttpRequest.get_signed_cookie(key, default=RAISE_ERROR, salt='', max_age=None)
返回签名过的Cookie 对应的值,如果签名不再合法则返回django.core.signing.BadSignature。
如果提供 default 参数,将不会引发异常并返回 default 的值。
可选参数salt 可以用来对安全密钥强力攻击提供额外的保护。max_age 参数用于检查Cookie 对应的时间戳以确保Cookie 的时间不会超过max_age 秒。
复制代码
>>> request.get_signed_cookie('name')
'Tony'
>>> request.get_signed_cookie('name', salt='name-salt')
'Tony' # 假设在设置cookie的时候使用的是相同的salt
>>> request.get_signed_cookie('non-existing-cookie')
...
KeyError: 'non-existing-cookie' # 没有相应的键时触发异常
>>> request.get_signed_cookie('non-existing-cookie', False)
False
>>> request.get_signed_cookie('cookie-that-was-tampered-with')
...
BadSignature: ...
>>> request.get_signed_cookie('name', max_age=60)
...
SignatureExpired: Signature age 1677.3839159 > 60 seconds
>>> request.get_signed_cookie('name', False, max_age=60)
False
复制代码
5.HttpRequest.is_secure()
如果请求时是安全的,则返回True;即请求通是过 HTTPS 发起的。
6.HttpRequest.is_ajax()
如果请求是通过XMLHttpRequest 发起的,则返回True,方法是检查 HTTP_X_REQUESTED_WITH 相应的首部是否是字符串'XMLHttpRequest'。
大部分现代的 JavaScript 库都会发送这个头部。如果你编写自己的 XMLHttpRequest 调用(在浏览器端),你必须手工设置这个值来让 is_ajax() 可以工作。
如果一个响应需要根据请求是否是通过AJAX 发起的,并且你正在使用某种形式的缓存例如Django 的 cache middleware, 你应该使用 vary_on_headers('HTTP_X_REQUESTED_WITH') 装饰你的视图以让响应能够正确地缓存。
7.HttpRequest.read(size=None)
像文件一样读取请求报文的内容主体,同样的,还有以下方法可用。
HttpRequest.readline()
HttpRequest.readlines()
HttpRequest.xreadlines()
其行为和文件操作中的一样。
HttpRequest.__iter__():说明可以使用 for 的方式迭代文件的每一行。'''
View Code
注意:键值对的值是多个的时候,比如checkbox类型的input标签,select标签,需要用:
request.POST.getlist("hobby")
3.3 render 函数
render(request, template_name[, context])
结合一个给定的模板和一个给定的上下文字典,并返回一个渲染后的 HttpResponse 对象。
参数:
request: 用于生成响应的请求对象。
template_name:要使用的模板的完整名称,可选的参数
context:添加到模板上下文的一个字典。默认是一个空字典。如果字典中的某个值是可调用的,视图将在渲染模板之前调用它。
content_type:生成的文档要使用的MIME类型。默认为DEFAULT_CONTENT_TYPE 设置的值。
status:响应的状态码。默认为200。
3.4 redirect 函数
参数可以是:
l 一个模型:将调用模型的get_absolute_url() 函数
l 一个视图,可以带有参数:将使用urlresolvers.reverse 来反向解析名称
l 一个绝对的或相对的URL,将原封不动的作为重定向的位置。
默认返回一个临时的重定向;传递permanent=True 可以返回一个永久的重定向。
示例:
你可以用多种方式使用redirect() 函数。
3.4.1 传递一个对象
将调用get_absolute_url() 方法来获取重定向的URL:
from django.shortcuts importredirectdefmy_view(request):
...
object=MyModel.objects.get(...)return redirect(object)
View Code
3.4.2 传递一个视图的名称
可以带有位置参数和关键字参数;将使用reverse() 方法反向解析URL:
defmy_view(request):
...return redirect('some-view-name', foo='bar')
3.4.3 传递要重定向的一个硬编码的URL
defmy_view(request):
...return redirect('/some/url/')
也可以是一个完整的URL:
defmy_view(request):
...return redirect('http://example.com/')
默认情况下,redirect() 返回一个临时重定向。以上所有的形式都接收一个permanent 参数;如果设置为True,将返回一个永久的重定向:
defmy_view(request):
...
object=MyModel.objects.get(...)return redirect(object, permanent=True)
3.4.4 跳转(重定向)应用
-----------------------------------url.py
url(r"login", views.login),
url(r"yuan_back", views.yuan_back),-----------------------------------views.pydeflogin(req):if req.method=="POST":if 1:#return redirect("/yuan_back/")
name="yuanhao"
return render(req,"my backend.html",locals())return render(req,"login.html",locals())defyuan_back(req):
name="苑昊"
return render(req,"my backend.html",locals())-----------------------------------login.html
姓名
性别
邮箱
-----------------------------------my backend.html
用户{{ name }}你好
View Code
redirect关键点:两次请求过程,掌握流程。
注意:render和redirect的区别:
1、 if 页面需要模板语言渲染,需要的将数据库的数据加载到html,那么render方法则不会显示这一部分。
2、 the most important: url没有跳转到/yuan_back/,而是还在/login/,所以当刷新后又得重新登录。
第4章 模板层(template)
python的模板:HTML代码+模板语法
模版包括在使用时会被值替换掉的变量,和控制模版逻辑的标签。
defcurrent_time(req):#================================原始的视图函数
#import datetime
#now=datetime.datetime.now()
#html="
现在时刻:%s.
" %now#================================django模板修改的视图函数
#from django.template import Template,Context
#now=datetime.datetime.now()
#t=Template('
现在时刻是:{{current_date}}
')##t=get_template('current_datetime.html')
#c=Context({'current_date':str(now)})
#html=t.render(c)
#
#return HttpResponse(html)
#另一种写法(推荐)
importdatetime
now=datetime.datetime.now()return render(req, 'current_datetime.html', {'current_date':str(now)[:19]})
View Code
模板语法:目的是将变量(数据库的内容)如何巧妙的嵌入到html页面中(就不用之前我们用的字符串拼接了)
在 Django 模板中遍历复杂数据结构的关键是句点字符 .
语法:
{{var_name}}
4.1 示例
#####################view.py##############################
defindex(request):importdatetime
s="hello"l=[111,222,333] #列表
dic={"name":"yuan","age":18} #字典
date= datetime.date(1993, 5, 2) #日期对象
classPerson(object):def __init__(self,name):
self.name=name
person_yuan=Person("yuan") #自定义类对象
person_egon=Person("egon")
person_alex=Person("alex")
person_list=[person_yuan,person_egon,person_alex]return render(request,"index.html",{"l":l,"dic":dic,"date":date,"person_list":person_list})
view.py
template:
{{s}}
列表:{{ l.0 }}
列表:{{ l.2 }}
字典:{{ dic.name }}
日期:{{ date.year }}
类对象列表:{{ person_list.0.name }}
View Code
注意:句点符也可以用来引用对象的方法(无参数方法)。
字典:{{ dic.name.upper }}
4.2 小练习
需求:当用户输入一个URL: http://127.0.0.1:8080/timer,就返回给用户一个当前的时间
基本流程
4.2.1 对urls进行一个设置
4.2.2 views.py
4.2.3 在template里创建一个timer.html页面
第5章 Django框架之第二篇反向解析
5.1 知识点回顾
5.1.1 MTV模型
model:模型,和数据库相关的
template:模板,存放html文件,模板语法(目的是将变量如何巧妙的嵌入到HTML页面中)。
views:视图函数
另加urls:url路径与视图函数的映射关系,,可以不是一一对应的。
5.1.2 相关的一些命令
创建一个Django项目:django-admin startproject projectname
创建一个项目下的应用:python3 manage.py startapp appname
运行:python3 manage.py runserver IP PORT
5.1.3 url配置(URLconf)urls.py
功能:建立起url与视图函数的映射关系
url(正则表达式(规则),视图函数,[可选参数])
url:http://127.0.0.1:8080/blog/articles/2003/05?a=1&b=2
匹配字符串:用户输入的url对应的路径 /blog/articles/2003/05
注意:
(1)出现覆盖现象的情况,也就是匹配规则冲突的时候,匹配第一个url
(2)无名分组:url(r'^articles/(\d{4})/(\d{2})$', views.year_month), # year(requset,1990,12) 按位置传参数
(3)有名分组:url(r'^articles/(?P\d{4})/(?P\d{2})$', views.year_month), # year(requset,year=1990,month=12) 按位置传参数
(4)url分发:url(r'^blog/',include('blog.urls'))
5.2 视图函数的补充
5.2.1 视图函数
1、视图函数:一定是要包含两个对象的(render源码里面有HttpResponse对象)
request对象:-----》所有的请求信息
HttpResponse:-----》响应的内容(字符串)
5.2.2 get请求发送数据
2、get请求发送数据:http://127.0.0.1:8000/login.html?user=asd&pwd=asd
重点:request里包含哪些数据1、request.GET: GET请求的数据,如果没有数据是一个空字典 {}2、request.POST:POST请求的数据 ,如果没有数据是一个空字典 {}3、request.method:请求方式:GET 或 POST4、请求某个键下多个值时:request.POST.getlist("hobby")5、 request.path : 请求路径(只会拿到路径,不拿数据)
请求url:http://127.0.0.1:8000/index.html/23?a=1path:request.path:/index.html/23
6、request.get_full_path():请求路径(路径和数据都会拿到)
请求url:http://127.0.0.1:8000/index.html/23?a=1request.get_full_path():/index.html/23?a=1
5.3 render函数和redirect函数的区别
render:只会返回页面内容,但是未发送第二次请求
redirect:发挥了第二次请求,url更新
具体实例说明如下:
5.3.1 render:
5.3.2 redirect:
5.4 反向解析
5.4.1 介绍
在使用Django 项目时,一个常见的需求是获得URL 的最终形式,以用于嵌入到生成的内容中(视图中和显示给用户的URL等)或者用于处理服务器端的导航(重定向等)。
人们强烈希望不要硬编码这些URL(费力、不可扩展且容易产生错误)或者设计一种与URLconf 毫不相关的专门的URL 生成机制,因为这样容易导致一定程度上产生过期的URL。
换句话讲,需要的是一个DRY 机制。除了其它有点,它还允许设计的URL 可以自动更新而不用遍历项目的源代码来搜索并替换过期的URL。
获取一个URL 最开始想到的信息是处理它视图的标识(例如名字),查找正确的URL 的其它必要的信息有视图参数的类型(位置参数、关键字参数)和值。
Django 提供一个办法是让URL 映射是URL 设计唯一的地方。你填充你的URLconf,然后可以双向使用它:
l 根据用户/浏览器发起的URL 请求,它调用正确的Django 视图,并从URL 中提取它的参数需要的值。
l 根据Django 视图的标识和将要传递给它的参数的值,获取与之关联的URL。
第一种方式是我们在前面的章节中一直讨论的用法。第二种方式叫做反向解析URL、反向URL 匹配、反向URL 查询或者简单的URL 反查。
在需要URL 的地方,对于不同层级,Django 提供不同的工具用于URL 反查:
l 在模板中:使用url 模板标签。
l 在Python 代码中:使用django.core.urlresolvers.reverse() 函数。
l 在更高层的与处理Django 模型实例相关的代码中:使用get_absolute_url() 方法。
5.4.2 例子:
考虑下面的URLconf:
from django.conf.urls importurlfrom . importviews
urlpatterns=[#...
url(r'^articles/([0-9]{4})/$', views.year_archive, name='news-year-archive'),#...
]
View Code
根据这里的设计,某一年nnnn对应的归档的URL是/articles/nnnn/。
你可以在模板的代码中使用下面的方法获得它们:
- {% for yearvar in year_list %}
- {{ yearvar }} Archive{% endfor %}
View Code
在Python 代码中,这样使用:
from django.core.urlresolvers importreversefrom django.http importHttpResponseRedirectdefredirect_to_year(request):#...
year= 2006
#...
return HttpResponseRedirect(reverse('news-year-archive', args=(year,)))
View Code
如果出于某种原因决定按年归档文章发布的URL应该调整一下,那么你将只需要修改URLconf 中的内容。
在某些场景中,一个视图是通用的,所以在URL 和视图之间存在多对一的关系。对于这些情况,当反查URL 时,只有视图的名字还不够。
5.4.3 例子分析
例子中:
分析:想我们一开始写的硬编码,也就是吧action要跳转的路径写死了。但是像淘宝,天猫等都会经常更新新东西,,那么你的页面上的url路径也会时不时的变化。但是如果有特别多的商品,那么你就得去服务端一个一个的改,这样显得很麻烦,那么有没有一种机制帮我们解决问题呢?那就按照我下面的办法解决。就把url路径写活了。
1、首先给url起一个别名。
2、然后在login.html中写上{% url ‘别名’’ %} ,如果在页面中点击查看元素,它会变成login.html,,,当然我的是分发了,,就会变成test/login.html
3、这样你就可以修改你的正则了,,因为他是按照别名走的,不会影响。
urls.py
login.html
查看元素的结果:
这样的好处是:无论你怎么改你要匹配的url,只要你写上了别名。在html实现了模板语法,就会去找别名对应的那个url,以后不管你怎么改url都没事,就写活了,就不像一开始写的硬编码了。
第6章 Django框架之第三篇模板语法
6.1 什么是模板?
只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板。
6.2 模板语法分类
6.2.1 模板语法之变量:语法为 {{ }}:
在 Django 模板中遍历复杂数据结构的关键是句点字符 .(也就是点)
views.py
defindex(request):
name= "hello haiyan"i= 200l= [11,22,33,44,55]
d= {"name":"haiyan","age":20}class People(object): #继承元类
def __init__(self,name,age):
self.name=name
self.age=agedef __str__(self):return self.name+str(self.age)defdream(self):return "你有梦想吗?"
#实例化
person_egon= People("egon",10)
person_dada= People("dada",34)
person_susan= People("susan",34)
person_list=[person_dada,person_egon,person_susan]return render(request,"index.html",
{"name":name,"i":i,"l":l,"d":d, #键对应的是模板里的名字。值对应的是上面定义的变量
"person_egon":person_egon,"person_dada":person_dada,"person_list":person_list,
}
)#return render(request,"index.html",locals())
#用locals()可以不用写上面的render了。不过用locals(),views里面用什么名。模板里面就得用什么名
#locals()局部的:用了locals就相当于都得按照上面的那样
View Code
template/index.html
变量{{ z }}:深度查询
{{ name }}
{{ i }}
{{ l }}
{{ d }}
{{ l.0 }}------》取单个值可通过句点符(也就是点)
{{ l.4 }}
{{ d.name }}
{{ d.age }}-----》字典也可以根据句点符取值,一个点就搞定了。
然而在前端页面中是看不到你的模板语法的,当你点击审查元素的
时候,你就会发现,偷偷的换过来了
{{ person_dada.name }}
{{ person_egon.age }}
{{ person_dada.dream }}
{{ person_list.2 }}
{{ person_list.1.name }}
View Code
注意:句点符也可以用来引用对象的方法(无参数方法)。
字典:{{ dic.name.upper }}
6.2.2 模板语法之标签:语法为 {% tag %}:
标签看起来像是这样的: {% tag %}。标签比变量更加复杂:一些在输出中创建文本,一些通过循环或逻辑来控制流程,一些加载其后的变量将使用到的额外信息到模版中。
一些标签需要开始和结束标签 (例如{% tag %} ...标签 内容 ... {% endtag %})。
1、for标签(注:循环序号可以通过{{forloop}}显示)
循环取值1
{% for item in person_list %}
{{ item.name }},{{ item.age }}
{% endfor %}循环取值2:倒序
{% for item in person_list reversed %}
{{ forloop.counter }}----->{{ item.name }},{{ item.age }}
{{ forloop.counter0 }}----->{{ item.name }},{{ item.age }}
{{ forloop.revcounter }}----->{{ item.name }},{{ item.age }}
{% endfor %}循环取值3:字典
{% for k,v in d.items %}
{{ k }},{{ v}}
{% endfor %}View Code
2、for....empty:for标签带有一个可选的{% empty %}从句,以便在给出的组是空的或者没有被找到时,可以有所操作。
{% for person in person_list %}
{{ person.name }}
{% empty %}sorry,no person here
{% endfor %}View Code
3、if标签:{% if %}会对一个变量求值,如果它的值是“True”(存在、不为空、且不是boolean类型的false值),对应的内容块会输出。
{% if i > 300 %}
大于{{ i }}
{% elif i == 200 %}等于{{ i }}
{% else %}小于{{ i }}
{% endif %}View Code
4、with:使用一个简单地名字缓存一个复杂的变量,当你需要使用一个“昂贵的”方法(比如访问数据库)很多次的时候是非常有用的
{% with total=business.employees.count %}
{{ total }} employee{{ total|pluralize }}
{% endwith %}
{{ person_list.2.name }}
{% with name=person_list.2.name %}{{ name }}
{% endwith %}View Code
5、csrf_token:这个标签用于跨站请求伪造保护
提交数据的时候就会做安全机制,当你点击提交的时候会出现一个forbbiddon的错误,就是用setting配置里的scrf做安全机制的,那么我们可以把它给注释了,或者在form表单下面添加一个{% csrf_token %} ,这才是真正解决的办法,注释不是解决的办法
scrf_token
{% csrf_token %}View Code
6.2.3 模板语法之过滤器:语法 {{obj|filter__name:param}}
1、default:如果一个变量是false或者为空,使用给定的默认值。否则,使用变量的值。例如:
default过滤器:{{ li|default:"如果显示为空,设置的解释性的内容" }}
2、length:返回值的长度。它对字符串和列表都起作用。例如:
{{ value|length }}
如果 value 是 ['a', 'b', 'c', 'd'],那么输出是 4。
3、filesizeformat:将值格式化为一个“人类可读的”文件尺寸(例如'13 KB','4.1 MB','102 bytes',等等)。例如:
{{ value|filesizeformat }}
如果 value 是 123456789,输出将会是 117.7 MB。
4、date:如果 value=datetime.datetime.now()
{{ value|date:"Y-m-d" }}
5、slice:切片
如果 value="hello world"
{{ value|slice:"2:-1" }}
6、truncatechars截断
如果字符串字符多于指定的字符数量,那么会被截断。截断的字符串将以可翻译的省略号序列(“...”)结尾。
参数:要截断的字符数
例如:
截断字符:{{ content|truncatechars:20 }}
截断单词:{{ content|truncatewords:4 }}
如果content是“I am is haiyan,how are you asd df dfgfdgdg?
输出结果: 截断字符:I am is haiyan,ho...
输出结果 :截断单词:I am is haiyan,how ...
7、safe
Django的模板中会对HTML标签和JS等语法标签进行自动转义,原因显而易见,这样是为了安全。但是有的时候我们可能不希望这些HTML元素被转义,比如我们做一个内容管理系统,后台添加的文章中是经过修饰的,这些修饰可能是通过一个类似于FCKeditor编辑加注了HTML修饰符的文本,如果自动转义的话显示的就是保护HTML标签的源文件。为了在Django中关闭HTML的自动转义有两种方式,如果是一个单独的变量我们可以通过过滤器“|safe”的方式告诉Django这段代码是安全的不必转义。比如:
value="点击"{{ value|safe}}
{{ label }}
{{ label|safe }}
这里简单介绍一些常用的模板的过滤器,更多详见
6.2.4 自定义标签和过滤器
1、在settings中的INSTALLED_APPS配置当前app,不然django无法找到自定义的simple_tag.
2、在app中创建templatetags模块(模块名只能是templatetags)
3、在templatetags里面创建任意.py文件,
如:my_tags.py
from django importtemplatefrom django.utils.safestring importmark_safe
register= template.Library() #register的名字是固定的,不可改变
@register.filter 过滤器defmulti(x,y):return x*y
@register.simple_tag 标签defmultitag(x,y,z):return x*y*z
@register.simple_tag 标签defmy_input(id,arg):
result= "" %(id,arg,)return mark_safe(result)
View Code
4、在使用自定义simple_tag和filter的html文件中导入之前创建的my_tags.py
{% load my_tags %}
5、使用simple_tag和filter(如何调用)
过滤器: {{ var|filter_name:参数 }} # 参数只能是两个,一个参数是变量var ,一个是参数是后面的那个参数
标签: {% simple_tag 参数1 参数2 ... %}
-------------------------------.html
{% load xxx %}#num=12
{{ num|multi:2 }} #24
{{ num|multi:"[22,333,4444]" }} 相当于复制了,吧[22,333,4444]乘了num遍
{% multitag 2 5 6 %} 参数不限,但不能放在if for语句中 {% simple_tag_multi num 5 %}
View Code
自定义过滤器函数的参数只能两个,可以进行逻辑判断
自定义标签无参数限制,不能进行逻辑判断
{% if i|multi:5 > 1000 %}
大于{{ i }}
{% else %}大于等于{{ i }}
{% endif %}View Code
第7章 Django框架之模板继承和静态文件配置
7.1 模板继承
目的是:减少代码的冗余
语法:
{% block classinfo %}
{% endblock %}
7.1.1 具体步骤
1、创建一个base.html文件,
2、把要显示的页面的内容写在这里面,也就是html要在浏览器显示的内容
3、在right里面写个盒子
{% block classinfo %}
{% endblock %}
在这里面写个空盒子,以后谁来扩展就在这个盒子里面添加相应的内容就行了
4、然后再创建一个.html文件,让这个继承base.html文件,
{% extends "base.html" %} #必须是在文件的第一行
在基板里面添加内容
{% block classinfo %}
首页
学生信息
{{ class_id }}班
{% endblock%}
5、也可以写好多盒子,
在left中写个盒子
{% block menu %}
I see you you
{% endblock %}
注意:
盒子里面可以有默认的内容,如果有默认的时候你不扩展就走默认的,如果你扩展了,就替换了,那么不替换直接追加可以嘛?可以的,那就用下面的方式。
{% block.super %}
例如:
{% block menu %}
{{ block.super }}
!!!
#先继承父类的,后插入数据{% endblock %}
View Code
7.1.2 注意项
1、模板继承围绕两点:继承和扩展
你有什么继承什么,
扩展的是盒子,
2、模板中设置的盒子越多越好,因为这样你想扩展的时候就容易了。我想扩展就扩展了。不扩展就不扩展了
3、为了更好的可读性,你也可以给你的 {% endblock %} 标签一个 名字 。例如:
{% block content %}
...
{% endblock content %}
4、如果你发现你自己在大量的模版中复制内容,那可能意味着你应该把内容移动到父模版中的一个 {% block %} 中。
最后,请注意你并不能在一个模版中定义多个相同名字的block标签。这个限制的存在是因为block标签的作用是“双向”的。这个意思是,block标签不仅提供了一个坑去填,它还在 _父模版_中定义了填坑的内容。如果在一个模版中有两个名字一样的 block 标签,模版的父模版将不知道使用哪个block的内容。
7.2 具体例子说明
7.2.1 urls.py
urlpatterns =[
url(r'^admin/', admin.site.urls),
url(r'^text/(\d+)', views.text),
]
7.2.2 views.py
from django.shortcuts importrender,redirect#Create your views here.
deftext(request,class_id):#班级变量
print(class_id) #拿到的是你在路径里输入的几就是几
#数据库查询
return render(request, "text.html", {"class_id": class_id})
View Code
7.2.3 templaite / base.html
height: 400px;
background-color: silver;
}
{#内容#}
View Code
7.2.4 template /text.py 继承上面的文件
{% extends "base.html" %}
{% block classinfo %}
学生信息
{{ class_id }}班级
{% endblock %}{#追加#}
{% block menu %}
{{ block.super }}学生信息{% endblock %}
View Code
7.3 静态文件配置
我们自己导入的一些包就叫做静态文件
1、在全局中先创建一个static的包,
2、在static里面导入我们的bootstrap,还是jquery等
3、然后在settings.py中加上一些配置
STATIC_URL = '/static/' #这个配置就相当于下面配置的别名,如果这里的名字修改了就按照这里的名字去导入
STATICFILES_DIRS=[
os.path.join(BASE_DIR,"static") #E:\day68\static 找到static路径
]
4、导入css,js,jquery
第8章 django之ORM数据库操作
8.1 ORM介绍
映射关系:
表名 --------------------》类名
字段--------------------》属性
表记录-----------------》类实例化对象
ORM的两大功能:
操作表:
- 创建表
- 修改表
- 删除表
操作数据行:
- 增删改查
ORM利用pymysql第三方工具链接数据库
Django没办法帮我们创建数据库,只能我们创建完之后告诉它,让django去链接
8.2 创建表之前的准备工作
一、自己创建数据库
二、在settings里面配置mysql数据库链接
sqlite3------改为mysql
#修改django默认的数据库的sqlite3为mysql
DATABASES={'default': {'ENGINE': 'django.db.backends.mysql', #通过这个去链接mysql
'NAME': 'djangotsgl','USER':'root','PASSWORD':'123456','HOST':'localhost','PORT':'3306',
}
}
View Code
这样写上以后django会默认的就去链接数据库,这时你会看到报错了,那么解决的办法就是下面的这样。
三、app01中的--init--文件
importpymysql
pymysql.install_as_MySQLdb()
四、创建数据库表
models.py
class Book(models.Model): #必须要继承的
nid= models.AutoField(primary_key=True) #自增id(可以不写,默认会有自增id)
title= models.CharField(max_length=32)
publishDdata= models.DateField() #出版日期
author= models.CharField(max_length=32)
price= models.DecimalField(max_digits=5,decimal_places=2) #一共5位,保留两位小数
View Code
执行命令创建:(需要记住的!!!)
python3 manage.py makemigrations 创建脚本
python3 manage.py migrate 迁移
8.3 具体例子实现
8.3.1 model.py
8.3.2 urls.py
8.3.3 views.py
8.3.4 template /index.html
#######################图片内容具体#######################
Titlemargin-top: 50px;
}
图片内容具体
8.4 查看数据库的sql语句
五、查看数据库的sql语句(加在settings.py)
查看数据库执行代码
LOGGING={'version': 1,'disable_existing_loggers': False,'handlers': {'console':{'level':'DEBUG','class':'logging.StreamHandler',
},
},'loggers': {'django.db.backends': {'handlers': ['console'],'propagate': True,'level':'DEBUG',
},
}
}
View Code
第9章 Django之ORM数据库操作注意细节
9.1 多对多的正反向查询
classClass(models.Model):
name= models.CharField(max_length=32,verbose_name="班级名")
course= models.CharField(verbose_name="课程",max_length=32)def __str__(self):returnself.nameclassTeacher(models.Model):
name= models.CharField(max_length=23,verbose_name="姓名")
classes= models.ManyToManyField(verbose_name="所属班级",to="Class")def __str__(self):return self.name
View Code
9.1.1 题目1:查找娜娜老师所带的班级
#方式一:基于对象的查找
obj= models.Teacher.objects.filter(name="娜娜").first()print(obj.classes.all())print("娜娜老师带的班级",obj.classes.values("name"))#方式二:基于双下划线的查找
obj_cls= models.Teacher.objects.filter(name="娜娜").values("classes__name")print("娜娜老师带的班级",obj_cls)
View Code
注意:要说明的是多对多的查询用.all,查单个的时候用.values或者values_list,不要用obj.classes.name,这样查到的会是None,反向查询也是如此。我就是犯了这样的错,引以为戒。。
总结:不管是一对多,还是多对多,要是查询多得一方就得用all()
运行结果截图:
表结构:
from django.db importmodels#Create your models here.
#一个学生有一个班级,一个班级可以有好多学生,所以是
#一对多的关系,关联字段放在多的一方
classStudent(models.Model):
name= models.CharField(max_length=32,verbose_name="姓名")
age= models.IntegerField(verbose_name="年龄")
classes= models.ForeignKey(to="Class",verbose_name="所属班级")def __str__(self):returnself.nameclassClass(models.Model):
name= models.CharField(max_length=32,verbose_name="班级名")
course= models.CharField(verbose_name="课程",max_length=32)def __str__(self):returnself.nameclassTeacher(models.Model):
name= models.CharField(max_length=23,verbose_name="姓名")
classes= models.ManyToManyField(verbose_name="所属班级",to="Class")def __str__(self):returnself.name
View Code
9.1.2 查询海燕在那个班级
#方式一:
print("海燕所在的班级",models.Student.objects.filter(name="海燕").values("classes__name"))#方式二:
obj_cls= models.Student.objects.filter(name="海燕").first()print("海燕所在的班级",obj_cls.classes.name)
View Code
9.1.3 查询海燕所在班的老师的姓名
print("海燕所在班的老师的姓名",models.Student.objects.filter(name="海燕").values("classes__teacher__name"))
9.1.4 查询软件测试151班的所有学生的姓名
print("软件测试151班的所有学生的姓名",models.Class.objects.filter(name="软件测试151").values("student__name"))
obj= models.Class.objects.filter(name="软件测试151").first()#print("软件测试151班的所有学生的姓名",obj.student_set.name) #这样打印的结果是None
print("软件测试151班的所有学生的姓名",obj.student_set.all().values("name"))
View Code
9.2 需要掌握的一个很重要的知识点
1、form表单中要用submit,如果用button切记要加上type,不然button默认的type是submit,会有影响
注册
提交
上面两种写法是对的,功能一样。提交如果写成这种,默认为submit,本来doValidation方法里有提交功能了,
再加上按钮也是提交功能,会提交两次。所以使用按钮时最好指定type类型。
View Code
第10章 django之数据库表的单表查询
10.1 添加表记录
对于单表有两种方式
#添加数据的两种方式
#方式一:实例化对象就是一条表记录
Frank_obj= models.Student(name ="海东",course="python",birth="2000-9-9",fenshu=80)
Frank_obj.save()#方式二:
models.Student.objects.create(name="海燕",course="python",birth="1995-5-9",fenshu=88)
View Code
10.2 查询表记录
10.2.1 查询相关API
#查询相关API
#1、all():查看所有
student_obj=models.Student.objects.all()print(student_obj) #打印的结果是QuerySet集合
#2、filter():可以实现且关系,但是或关系需要借助Q查询实现。。。
#查不到的时候不会报错
print(models.Student.objects.filter(name="Frank")) #查看名字是Frank的
print(models.Student.objects.filter(name="Frank",fenshu=80)) #查看名字是Frank的并且分数是80的
#3、get():如果找不到就会报错,如果有多个值,也会报错,只能拿有一个值的
print(models.Student.objects.get(name="Frank")) #拿到的是model对象
print(models.Student.objects.get(nid=2)) #拿到的是model对象
#4、exclude():排除条件
print( models.Student.objects.exclude(name="海东")) #查看除了名字是海东的信息
#5、values():是QuerySet的一个方法 (吧对象转换成字典的形式了,)
print(models.Student.objects.filter(name="海东").values("nid","course")) #查看名字为海东的编号和课程
#打印结果:
#6、values_list():是queryset的一个方法 (吧对象转成元组的形式了)
print(models.Student.objects.filter(name="海东").values_list("nid", "course"))#打印结果:< QuerySet[(2, 'python'), (24, 'python')] >
#7、order_by():排序
print(models.Student.objects.all().order_by("fenshu"))#8、reverse():倒序
print(models.Student.objects.all().reverse())#9、distinct():去重(只要结果里面有重复的)
print(models.Student.objects.filter(course="python").values("fenshu").distinct())#10、count():查看有几条记录
print(models.Student.objects.filter(name="海东").count())#11、first()
#12、last()
return render(request,"test.html",{"student_obj":student_obj})#13、esits:查看有没有记录,如果有返回True,没有返回False
#并不需要判断所有的数据,
#if models.Book.objects.all().exists():
View Code
10.2.2 双下划线之单表查询
models.Tb1.objects.filter(id__lt=10, id__gt=1) #获取id小于1 且 大于10的值
models.Tb1.objects.filter(id__in=[11, 22, 33]) #获取id等于11、22、33的数据
models.Tb1.objects.exclude(id__in=[11, 22, 33]) #not in
models.Tb1.objects.filter(name__contains="ven") #包括ven的
models.Tb1.objects.filter(name__icontains="ven") #icontains大小写不敏感
models.Tb1.objects.filter(id__range=[1, 2]) #范围bettwen and
startswith,istartswith, endswith, iendswith
View Code
对象可以调用自己的属性,用一个点就可以
还可以通过双下划线。。。
models.Book.objects.filter(price__gt=100) 价格大于100的书
models.Book.objects.filter(author__startwith= "张") 查看作者的名字是以张开头的
主键大于5的且小于2
price__gte=99大于等于
publishDate__year=2017,publishDate__month = 10 查看2017年10月份的数据
10.3 修改表记录
注意:
<1> 第二种方式修改不能用get的原因是:update是QuerySet对象的方法,get返回的是一个model对象,它没有update方法,而filter返回的是一个QuerySet对象(filter里面的条件可能有多个条件符合,比如name='alvin',可能有两个name='alvin'的行数据)。
<2>在“插入和更新数据”小节中,我们有提到模型的save()方法,这个方法会更新一行里的所有列。 而某些情况下,我们只需要更新行里的某几列。
此外,update()方法对于任何结果集(QuerySet)均有效,这意味着你可以同时更新多条记录update()方法会返回一个整型数值,表示受影响的记录条数。
注意,这里因为update返回的是一个整形,所以没法用query属性;对于每次创建一个对象,想显示对应的raw sql,需要在settings加上日志记录部分
10.4 删除表记录
删除方法就是 delete()。它运行时立即删除对象而不返回任何值。例如:e.delete()
defdelstudent(request,id):#删除数据
models.Student.objects.filter(nid=id).delete()return redirect("/test/")
你也可以一次性删除多个对象。每个 QuerySet 都有一个 delete() 方法,它一次性删除 QuerySet 中所有的对象。
例如,下面的代码将删除 pub_date 是2005年的 Entry 对象:
Entry.objects.filter(pub_date__year=2005).delete()
要牢记这一点:无论在什么情况下,QuerySet 中的 delete() 方法都只使用一条 SQL 语句一次性删除所有对象,而并不是分别删除每个对象。如果你想使用在 model 中自定义的 delete() 方法,就要自行调用每个对象的delete 方法。(例如,遍历 QuerySet,在每个对象上调用 delete()方法),而不是使用 QuerySet 中的 delete()方法。
在 Django 删除对象时,会模仿 SQL 约束 ON DELETE CASCADE 的行为,换句话说,删除一个对象时也会删除与它相关联的外键对象。例如:
b = Blog.objects.get(pk=1)#This will delete the Blog and all of its Entry objects.
b.delete()
要注意的是: delete() 方法是 QuerySet 上的方法,但并不适用于 Manager 本身。这是一种保护机制,是为了避免意外地调用 Entry.objects.delete() 方法导致 所有的 记录被误删除。如果你确认要删除所有的对象,那么你必须显式地调用:
Entry.objects.all().delete()
10.5 编辑表格中的内容的涉及到的语法
编辑操作涉及到的语法
分析:1、点击编辑,让跳转到另一个页面,拿到我点击的那一行
两种取id值的方式
方式一:
利用数据传参数(作为数据参数传过去了) #相当于发了一个键值对
url里面就不用写匹配的路径了,
id= request.GET.get("book_id") #取值
方式二:
利用路径传参,得在url里面加上(\d+),就得给函数传个参数,无名分组从参数里面取值
2、拿到id,然后在做筛选
id= request.GET.get("book_id")
book_obj= models.Book.objects.filter(nid=id) #拿到的是一个列表对象
注意:1.取[0]就拿到对象了,,然后对象.属性就可以取到值了2.用get,你取出来的数据必须只有一条的时候,,如果有多条用get就会报错,,,但是用get就不用加[0]了
book_obj= models.Book.objects.filter(nid=id)[0]3、当点击编辑的时候怎么让input框里显示文本内容
value= "{{book_obj.title}}"
4、改完数据后重新提交
当提交的时候走action...../edit/}
隐藏一个input,判断post的时候:
修改数据
方式一:save(这种方式效率是非常低的,不推荐使用,了解就行了)
修改的前提是先取(拿到要编辑的id值)
id= request.POST.get("book_id")
bk_obj= models.Book.objects.filter(nid=id)[0]
bk_obj.title= "hhhhhh" #这是写死了,不能都像这样写死了
bk_obj.save() 只要是用对象的这种都要.save
方式二:update
title= request.POST.get("title")
models.Book.objects.filter(nid=id).update(title=title,......)
跳转到index
如果是post请求的时候怎么找到id呢,
一、如果是数据传参:(也就是get请求的时候)
可以通过一个隐藏的input框,给这个框给一个name属性,value属性。通过request.POST.get("键"),,就可以得到id的值
二、如果是路径传参
可以通过传参的形式,当正则表达式写一个(\d+)的时候,就给函数传一个id,可通过这个id知道id.
View Code
第11章 作业
作业要求,在Django里面实现一个简单的表单数据查询、新增、修改、删除等。