为啥 Django 中的一些包含需要字符串,而另一些需要变量名?

Posted

技术标签:

【中文标题】为啥 Django 中的一些包含需要字符串,而另一些需要变量名?【英文标题】:Why do some includes in Django need strings, and others variable names?为什么 Django 中的一些包含需要字符串,而另一些需要变量名? 【发布时间】:2015-06-26 12:15:00 【问题描述】:

参考Django Book, chapter 3:

from django.conf.urls import patterns, include, url

# Uncomment the next two lines to enable the admin:
# from django.contrib import admin
# admin.autodiscover()

urlpatterns = patterns('',
    # Examples:
    # url(r'^$', 'mysite.views.home', name='home'),
    # url(r'^mysite/', include('mysite.foo.urls')),
    # url(r'^admin/doc/', include('django.contrib.admindocs.urls')),
    # url(r'^admin/', include(admin.site.urls)),
)

是什么决定了为什么将一个事物作为字符串包含,而将另一个事物作为常规变量包含在内?为什么admin.site.urls 而不是'admin.site.urls'?所有其他 includes 都包含为字符串...我在这里看不到任何逻辑模式。

【问题讨论】:

【参考方案1】:

如果你传递给 include() 一个 url() 实例列表,那么你不要使用字符串(参见 Include() docs 和 Including other URLconfs docs)。在您的管理 url 示例中,admin.site.urls 指的是 url 实例列表。参见source code 中的get_urls 方法(它提供了admin.site.urls 引用的url 实例列表)。

def get_urls(self):
    ...
    # Admin-site-wide views.
    urlpatterns = [
        url(r'^$', wrap(self.index), name='index'),
        url(r'^login/$', self.login, name='login'),
        url(r'^logout/$', wrap(self.logout), name='logout'),
        url(r'^password_change/$', wrap(self.password_change, cacheable=True), name='password_change'),
        url(r'^password_change/done/$', wrap(self.password_change_done, cacheable=True),
            name='password_change_done'),
        url(r'^jsi18n/$', wrap(self.i18n_javascript, cacheable=True), name='jsi18n'),
        url(r'^r/(?P<content_type_id>\d+)/(?P<object_id>.+)/$', wrap(contenttype_views.shortcut),
            name='view_on_site'),
    ]

【讨论】:

如果我能放两个绿色箭头,我会给你一个【参考方案2】:

首先,第一个模式('mysite.views.home' -> 视图函数)在 1.8 中已被弃用:它导致了各种麻烦。

至于其余的,它通常都适用。 'mysite.foo.urls' 被解析为在模块mysite.foo.urls 中包含模式,但from mysite.foo import urls as foo_urls; include(foo_urls) 也可以工作。基于字符串的导入主要是尚未删除的历史工件,但它很方便并且没有任何真正的缺点,因为模块会立即导入(因此,任何ImportError 都可以轻松追溯到 url 配置)。

admin.site.urls 不同,因为admin.site.urls 不是一个模块,但site 是一个对象,urls 是一个属性。出于这个原因,admin.site.urls 的基于字符串的导入将不起作用,您必须使用第二种方法。

最后一点,Django Book 开头的警告说明它已经过时了,但已经过时了。更多最新的资源,例如官方文档(我所知道的最好的官方文档之一),将是可取的。

【讨论】:

在 dj 1.8 之后你会怎么做才能调用那里的视图?导入视图并将其命名为 url(r'^$', view_but_not_as_string, name='home')? 不包括:from mysite.views import home; ...url(r'^$', home, name='home'),...include()只是添加嵌套在当前模式中的其他模式。【参考方案3】:

在 Django 1.8 文档中,include() 可以将模块作为参数OR 模块名称

https://docs.djangoproject.com/en/1.8/ref/urls/#include

线

from django.contrib import admin

导入管理模块。因此,它可以直接用作包含的参数。其他 include() 方法具有命名其他 URLconf 的字符串参数。

【讨论】:

【参考方案4】:

您必须将带有字符串的路径放入函数 include()。但是 admin.site.urls 就像一个字符串,因为在代码的开头,你有一个 import

【讨论】:

【参考方案5】:

你说得对,这有点令人困惑。他们在 Django 1.8 中改变了这一点,所以现在他们只支持包含普通变量。

在Django 1.8 release note 中了解更多信息。

【讨论】:

提供的链接用于引用视图,而不是一般的 URLconfs

以上是关于为啥 Django 中的一些包含需要字符串,而另一些需要变量名?的主要内容,如果未能解决你的问题,请参考以下文章

为啥 iloc() 的一种使用会给出 SettingWithCopyWarning,而另一种则不会?

为啥有些参数是使用预定义的路由添加的,而另一些则放在查询字符串中?

Django Rest Framework 中的模型翻译

为啥有些字典键在从调试器打印时有引号而另一些没有?

Django - 为啥视图中的“默认值”不起作用?

为啥 django sqlite3 数据库在一台机器上与另一台机器上的工作方式不同?