最近在看django的官方文档,跟着文档敲第一个django应用
刚开始看到第一部分关于urls路由的设置时,脑子里冒出来许多问号,一时没转过弯来
它到底是咋转发路由的?
如何通过我配置的路径来定位到指定视图的?
然后自己就反复试验了多次,又仔细看了文档中的描述(我的蹩脚英语不足以支撑看原版英文文档,看的英译汉,所以翻译后有些字眼确实不好理解,没内个味道了...)
OK,闲话到此为止,继续往下看 首先,我的django项目层级如下
mysite/urls.py 是根 URLconf 文件
polls/urls.py 是一个应用下的 URLconf 文件
在 polls/urls.py 中添加了如下路由
from django.urls import path
from . import views
urlpatterns = [
path('', views.index, name='index'),
]
然后在mysite/urls.py文件的urlpatterns列表里插入一个include()
from django.contrib import admin
from django.urls import include, path
urlpatterns = [
path('polls/', include('polls.urls')),
path('admin/', admin.site.urls),
]
第一种情况
官方文档有这样一句话:
每当Django遇到include()时
,它都会截断直到该时间点匹配的URL的任何部分,并将剩余的字符串发送到包含的URLconf中以进行进一步处理。
上面的这段话有2个关键点:
0.当在浏览器输入一段url时,它会首先根据 mysite/urls.py
文件中urlpatterns包含的path匹配
1.例如输入url:
http://127.0.0.1:8000/polls/
那么它会以这串url中的 polls/ 进行匹配
2.在 mysite/urls.py
文件中,发现 urlpatterns
内的第一个 path 就是polls/ ,匹配成功
3.截断这个url已经匹配的部分
所以 http://127.0.0.1:8000/polls/ 截断后只剩下空字符串 ''
4.接下来就把空字符串 '' 继续送给 include('polls.urls')
包含的 URLconf 文件进行处理
这里的话,就是送给投票应用 polls 自己的 URLconf 文件来继续匹配这个剩下的空字符串 ''
5.观察 polls/urls.py
中urlpatterns
包含的path
第一个 path 就是空字符串 '',匹配成功,所以最终这个url就映射到了这个空字符串对应的视图
如果访问 http://127.0.0.1:8000/polls/
等价于访问
http://127.0.0.1:8000/polls/ + 空字符串
第二种情况
如果上面的空字符串不好理解,可以再换个url
例如url为 http://127.0.0.1:8000/polls/test/
mysite/urls.py
内容如下
from django.contrib import admin
from django.urls import include, path
urlpatterns = [
path('polls/', include('polls.urls')),
path('admin/', admin.site.urls),
]
polls/urls.py
中内容如下
from django.urls import path
from . import views
urlpatterns = [
path('', views.index, name='index'),
path('test/', views.index2, name='index2'), # 添加了一个新路由,映射到index2视图
]
视图文件添加了一个index2()函数 polls/views.py
from django.shortcuts import render
from django.http import HttpResponse
# Create your views here.
def index(request):
return HttpResponse("hello, world")
def index2(request):
return HttpResponse("测试匹配路由test")
按照如下思路进行分析:
1、在浏览器输入url: http://127.0.0.1:8000/polls/test/
它还是先在 mysite/urls.py
文件中以后缀“polls/test/” 匹配路由,可以发现 polls/ 匹配成功
2、截断url后,剩下 test/ ,将其发送给投票应用 polls 的URLconf文件继续处理,也就是 polls/urls.py
3、观察 polls/urls.py
中 urlpatterns
包含的path
第一个path
是 '',匹配失败;
第二个path是 test/
,匹配成功,
所以这个url就映射到了其对应的视图,也就是 views.index2
访问url,结果如下,确实映射到了视图函数index2()对应的内容
第三种情况
一个django项目下可以有多个应用(app),上面的示例中只有一个应用polls
接下来再在这个项目新建一个应用,名称为blog 然后在blog下新建一个urls.py文件
在 blog/views.py
文件中添加一个视图
from django.shortcuts import render
from django.http import HttpResponse
# Create your views here.
def index(request):
return HttpResponse("hello, welcome to blog")
在 blog/urls.py
文件中添加path
from django.urls import path
from blog import views
urlpatterns = [
path('index/', views.index, name='index'),
]
然后在根urlconf文件,也就是mysite/urls.py
文件中指定blog.urls模块
from django.contrib import admin
from django.urls import path, include
urlpatterns = [
path('polls-admin/', admin.site.urls),
path('polls/', include('polls.urls')),
path('blog/', include('blog.urls')), # 指定blog.urls模块
]
这个时候如果在浏览器输入 http://127.0.0.1:8000/blog/index/
根据上面的经验,它会先定位到blog/,然后到blog下的urls文件中继续匹配 index/
如果匹配成功,则映射到 blog/views.py
下定义的index视图
访问结果如下
如果修改 blog/urls.py
文件,把path中的 index/ 改为 index2/
from django.urls import path
from blog import views
urlpatterns = [
path('index2/', views.index, name='index'),
]
这个时候再访问 http://127.0.0.1:8000/blog/index/,就会报错了
因为这个时候是 index2/ 关联 index 视图,
而 index/ 这个路径没有映射(关联)任何视图了,此时会报404