Django从 URL获取参数的几种方式

URL也就是网址了,是django所有视图函数的门面,通过它才能进入纷繁复杂的django世界,下面根据官方文档总结一下URL获取参数的集中方式

url配置的根节点是 ROOT_URLCONF,这个配置是在项目的settings.py文件里,一般当你创建完项目后这个配置就自动出现在settings文件里而它的值就是项目名.urls, 知道这个配置你就可以任意修改url配置的根路径。上面这是普遍的情况,如果使用了某些中间件,那么请求HttpRequest本身可能会带有urlconf参数,这时候我们设置的urlconf就会被覆盖。

找到ROOT_URLCONF后,django加载这个模块,查找其中的变量urlpatterns,这个变量是一个list数据类型,而list里元素的类型因版本不同会有差异,django版本2.0之前使用的是django.conf.urls.url()对象,2.0及以后使用的是django.urls.path()和 django.urls.re_path(),

在1.X时代,只有django.conf.urls.url(),形式如下:

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),
]

这种类型的urlpattern都是使用正则表达式匹配的,如果需要获取参数,可以把对应的正则表达式部分放到括号里,
比如请求/articles/2005/03/,就会匹配到上面第三条,因为有两个括号,所以就会有两个参数,实际调用的视图函数方式为:views.month_archive(request, '2005', '03'),这种类型的参数叫unamed group参数,在写视图函数的时候有两种写法:

#第一种
def month_archive(request,year_p,month_p):
      print(year_p,month_p)
      ...

#第二种
或者第二种可变参数形式:
def month_archive(request,*args):
     year=args[0]
     month=args[1]
     ...

这种参数也叫位置参数positional arguments
(注意:/articles/2005/3/ 这种请求不会匹配上面任一种url,因为位数不对)

上面这种叫unamed group,那必然有一种叫named group,这种匹配的参数叫关键字参数keyword arguments.
这种urlpattern写法为:

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),
]

可见相比于第一种,第二种在前面增加了?P<year>部分,这就可以给获取到的参数取个名字放在字典里了,
例如:/articles/2005/03/ 这种url调用的函数就是: views.month_archive(request, year='2005', month='03')
视图里获取这种参数的形式为:

#第一种
def month_archive(request,year,month):
    print(year,month)
    ...
   **注意这里的参数名year和month必须和url里定义的参数名一致,不然会报错,这和unamed group的位置参数不一样

#第二种
def month_archive(request,**kwargs):
    print(kwargs['year'],kwargs['month'])
    ...

上面这些是django 1.X版本的写法,到了2.X版本,django不仅提供了正则表达式匹配,还有常规的匹配,
具体来说就是把1.X里的url()匹配改成了re_path(),而普通的字符串匹配用的是path(),正则表达式匹配re_path()的用法和1.X里的url()一样,下面看一下path()匹配的用法示例:

from django.urls import path

from . import views

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),
]

可见新的写法看上去更加简练了,获取参数的形式变成了中括号<>,这里面的第一个参数叫转换器converter ,第二个参数就是变量名了,注意在1.X版本里获取的变量都是字符串类型,这在2.X版本你不同,2.X可以定义变量的类型,使用的就是转换器converter ,例如这里的int就是将获取的参数转化成数字型传给视图直接用,django内置的转换器有如下几种:

  • str - Matches any non-empty string, excluding the path separator, '/'. This is the default if a converter isn’t included in the expression.
  • int - Matches zero or any positive integer. Returns an <cite>int</cite>.
  • slug - Matches any slug string consisting of ASCII letters or numbers, plus the hyphen and underscore characters. For example,building-your-1st-django-site.
  • uuid - Matches a formatted UUID. To prevent multiple URLs from mapping to the same page, dashes must be included and letters must be lowercase. For example, 075194d3-6885-417e-a8a8-6c931e272f00. Returns a UUID instance.
  • path - Matches any non-empty string, including the path separator, '/'. This allows you to match against a complete URL path rather than just a segment of a URL path as with str.

django同样赋予了开发者自定义转换器的功能,这里不做介绍,详见https://docs.djangoproject.com/en/2.1/topics/http/urls/#registering-custom-path-converters

上面介绍的是从urlpattern里获取参数,是django内置实现的,还有一种是从HTTP请求里获取参数:
常见的是GET和POST请求,这里的参数被封装在请求request里,GET请求的数据在request.GET里,
POST的请求放在request.POST里
对于GET请求,一般参数直接出现在url里,
如domain/search/?q=haha ,可以用request.GET.get('q')或request,GET['q']获取到q参数,
POST请求的数据不会出现在url里。
一般我们获取参数使用request.GET.get('xx','')/request.POST.get('xx','')的形式,get函数的第二个参数是当没有xx参数时返回的默认值

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351