-
1、FBV和CBV
-
2、request对象和response对象
-
3、路由系统
1、FBV和CBV
1.FBV(function base view)基于函数的视图
2.CBC(class base view)基于类的视图
3.二者区别以下:
#FBV
def add_author(request):
if request.method == 'POST':
new_name = request.POST.get('add_name')
new_books = request.POST.getlist('books')
ret = models.Author.objects.create(name=new_name)
ret.book.set(new_books)
ret.save()
return redirect('/author_list/')
ret = models.Book.objects.all()
return render(request, 'add_author.html', {'book_list': ret})
#CBV
#导入
from django.views import View
#必须继承View
class AddAuthor(View):
#若是是get方法
def get(self , request):
pass
#若是是post方法
def post(self, request):
pass
url(r'^add_author/', views.AddAuthor.as_view()),
#必需要这样调用类的方法
4.结论:
二者各有优劣,类的逻辑更加清晰一点
2、request对象和reponse对象
1.request相关的经常使用值:
path_info 返回用户访问url,不包括域名和端口
method 请求中使用的HTTP方法的字符串表示,全大写表示。
GET 包含全部HTTP GET参数的类字典对象
POST 包含全部HTTP POST参数的类字典对象
body 请求体,byte类型 request.POST的数据就是从body里面提取到的
2.上传文件的简单实例:
<form action="/test/" method="
post"
enctype="multipart/form-data">
#后面这个必定要写
<input type="
file" name="file">
<input type="submit" name="上传">
</form>
def test(request):
if request.method == 'POST':
#从请求的FILES中获取上传文件的文件名,file为页面上type=files类型的input的name属性值
filename = request.FILES['file'].name
#在项目目录下新建一个文件
with open(filename, 'wb') as f:
#从上传的文件对象中一点一点读
for chunk in request.FILES['file'].chunks():
f.write(chunk)
return HttpResponse('上传成功')
而后在网页上上传文件就能够接收了
3.response相关:
a.HttpResponse
-->返回字符串
b.
render
-->返回一个html页面
c.redirect -->返回一个重定向(告诉浏览器访问另一个网站)
d.JsonResponse
最原始的写法:
加json.dumps():
def base(request):
data = {'name':'汤姆', 'age':18}
import json
return HttpResponse(
json.dumps(data))
#把data序列化称json格式的字符串
#页面返回的结果
{"name": "\u6c64\u59c6", "age": 18}
不加json.dumps():
def base(request):
data = {'name':'汤姆', 'age':18}
import json
return HttpResponse(data)
django封装后的写法:
def base(request):
data = {'name':'汤姆', 'age':18}
from django.http import JsonResponse
return JsonResponse(data)
#页面返回的结果
{"name": "\u6c64\u59c6", "age": 18}
可是,django封装的会出现一个问题:只能接收字典,不能接收列表,须要把安全监测关闭,以下:
from django.http import JsonResponse
return JsonResponse(data,
safe=False
)
3、路由系统
1.基本格式:
from django.conf.urls import url
urlpatterns = [
url(正则表达式, views视图函数,参数,别名),
]
django2.0版本中的路由系统已经替换成下面的写法:
from django.urls import path
urlpatterns = [
path('articles/2003/', views.special_case_2003),
path('articles/<int:year>/', views.year_archive),
path('articles/<int:year>/<int:month>/', views.month_archive),
path('articles/<int:year>/<int:month>/<slug:slug>/', views.article_detail),
]
2.参数说明:
正则表达式:一个正则表达式字符串
views视图函数:一个可调用对象,一般为一个视图函数或一个指定视图函数路径的字符串
参数:可选的要传递给视图函数的默认参数(字典形式)
别名:一个可选的name参数
3.基本配置:
from django.conf.urls import url
from . import views
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),
]
4.注意事项:
a. urlpatterns中的元素按照书写顺序从上往下逐一匹配正则表达式,
一旦匹配成功则再也不继续。
b. 若要从URL中捕获一个值,只须要在它周围放置一对
圆括号(分组匹配)。
c.
不须要添加一个前导的反斜杠,由于每一个URL 都有。例如,应该是^articles 而不是 ^/articles。
d. 每一个正则表达式前面的'r' 是可选的可是
建议加上。
5.url自动加/的问题:
# 是否开启URL访问地址后面不为/跳转至带有/的路径的配置项
# 把这行代码直接加载settings最后一行就ok了
APPEND_SLASH=True
Django settings.py配置文件中默认没有 APPEND_SLASH 这个参数,但 Django 默认这个参数为 APPEND_SLASH = True。 其做用就是自动在网址结尾加'/'。
其效果就是:
咱们定义了urls.py:
from django.conf.urls import url
from app01 import views
urlpatterns = [
url(r'^blog/$', views.blog),
]
访问 http://www.example.com/blog 时,默认将网址自动转换为 http://www.example/com/blog/ 。
若是在settings.py中设置了 APPEND_SLASH=False,此时咱们再请求 http://www.example.com/blog 时就会提示找不到页面。
6.分组命名匹配:
上面的示例使用简单的正则表达式分组匹配(经过圆括号)来
捕获URL中的值并以位置参数形式传递给视图。
在更高级的用法中,能够使用
分组命名匹配的正则表达式组来捕获URL中的值并以关键字参数形式传递给视图。
在Python的正则表达式中,分组命名正则表达式组的语法是
(?P<name>pattern),其中name是组的名称,pattern是要匹配的模式。
下面是以上URLconf 使用命名组的重写:
from django.conf.urls import url
from . import views
urlpatterns = [
url(r'^articles/
2003/$', views.special_case_2003),
url(r'^articles/(?P<year>[0-9]{4})/$', views.year_archive),
url(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$', views.month_archive),
url(r'^articles/
(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/(?P<day>[0-9]{2})/$', views.article_detail),
]
这个实现与前面的示例彻底相同,只有一个细微的差异:捕获的值做为
关键字参数而不是
位置参数传递给视图函数。
例如,针对url /articles/2017/12/至关于按如下方式调用视图函数:
views.month_archive(
request, year="2017", month="12")
在实际应用中,
使用分组命名匹配的方式可让你的URLconf 更加明晰且不容易产生参数顺序问题的错误,可是有些开发人员则认为分组命名组语法太丑陋、繁琐。
至于究竟应该使用哪种,你能够根据本身的喜爱来决定。
7.视图函数中的默认指定值:
# urls.py中
from django.conf.urls import url
from . import views
urlpatterns = [
# 默认返回第一页的blog
url
(
r'^blog/$', views.page),
# 返回你访问的具体页面的blog
url(
r'^blog/page(?P<num>[0-9]+)/$', views.page),
]
# views.py中,能够为num指定默认值
def page(request,
num="1"):
pass
在上面的例子中,两个URL模式指向相同的view - views.page - 可是第一个模式并无从URL中捕获任何东西。
若是第一个模式匹配上了,page()函数将使用其默认参数num=“1”,若是第二个模式匹配,page()将使用正则表达式捕获到的num值。
8.反向解析URL:
本质上就是给url匹配模式起别名,而后用过别名拿到具体的URL路径
a. 怎么起别名?
在url匹配模式中,定义name="别名"
url(r'^test/', views.test, name='test')
2. 如何使用?
1. 在模板语言里面使用:
{% url "别名" %} --> 获得具体的URL路径
2. 在视图中如何使用:
from django.urls import reverse
reverse("别名")
--> 获得具体的URL路径
3. 如何传参数?
1. 模板语言中:
{% url "别名" 2018 "nb" %}
2. 视图函数中
传位置参数:reverse("别名",
args=(2018, "nb"))
传关键字参数:reverse("别名"
kwargs={"year": 2018, "title": "nb"})
4. namespace
为了防止不一样的app下面的url匹配模式有重复的别名