7

我在 django 中有一个可以接受许多不同过滤器参数的视图,但它们都是可选的。如果我有 6 个可选过滤器,我真的必须为 6 个的每个组合编写 url,还是有办法定义 url 的哪些部分是可选的?

举一个只有 2 个过滤器的示例,我可以拥有所有这些 url 可能性:

/<city>/<state>/
/<city>/<state>/radius/<miles>/
/<city>/<state>/company/<company-name>/
/<city>/<state>/radius/<miles>/company/<company-name>/
/<city>/<state>/company/<company-name>/radius/<miles>/

所有这些 url 都指向同一个视图,唯一需要的参数是城市和州。使用 6 个过滤器,这变得难以管理。

做我想要实现的目标的最佳方法是什么?

4

4 回答 4

9

一种方法是让正则表达式将所有给定的过滤器作为单个字符串读取,然后将它们拆分为视图中的单个值。

我想出了以下网址:

(r'^(?P<city>[^/]+)/(?P<state>[^/]+)(?P<filters>(?:/[^/]+/[^/]+)*)/?$',
 'views.my_view'),

匹配所需的城市和州很容易。这filters部分有点复杂。内部部分 -(?:/[^/]+/[^/]+)*匹配表格中给出的过滤器/name/value。但是,*量词(像所有 Python 正则表达式量词一样)只返回找到的最后一个匹配项 - 所以如果 url/radius/80/company/mycompany/只是company/mycompany将被存储。相反,我们告诉它不要捕获单个值(?:在开头),并将它放在一个捕获块中,它将所有过滤器值存储为单个字符串。

视图逻辑相当简单。请注意,正则表达式只会匹配过滤器对 - 因此/company/mycompany/radius/不会被匹配。这意味着我们可以安全地假设我们有成对的值。我测试的视图如下:

def my_view(request, city, state, filters):
    # Split into a list ['name', 'value', 'name', 'value']. Note we remove the
    # first character of the string as it will be a slash.
    split = filters[1:].split('/')

    # Map into a dictionary {'name': 'value', 'name': 'value'}.
    filters = dict(zip(split[::2], split[1::2]))

    # Get the values you want - the second parameter is the default if none was
    # given in the URL. Note all entries in the dictionary are strings at this
    # point, so you will have to convert to the appropriate types if desired.
    radius = filters.get('radius', None)
    company = filters.get('company', None)

    # Then use the values as desired in your view.
    context = {
        'city': city,
        'state': state,
        'radius': radius,
        'company': company,
    }
    return render_to_response('my_view.html', context)

关于这一点需要注意两点。首先,它允许未知过滤器条目进入您的视图。例如,/fakefilter/somevalue有效。上面的视图代码忽略了这些,但您可能想向用户报告错误。如果是这样,请将获取值的代码更改为

radius = filters.pop('radius', None)
company = filters.pop('company', None)

字典中剩余的任何条目filters都是您可以抱怨的未知值。

其次,如果用户重复过滤,将使用最后一个值。例如,/radius/80/radius/50将半径设置为 50。如果要检测这一点,则需要在将值列表转换为字典之前对其进行扫描:

given = set()
for name in split[::2]:
    if name in given:
        # Repeated entry, complain to user or something.
    else:
        given.add(name)
于 2011-03-23T01:18:07.070 回答
8

这绝对是 GET 参数的用例。您的 urlconf 应该只是/city/state/,然后各种过滤器作为 GET 变量结束:

/city/state/?radius=5&company=google

现在,在您看来,您接受citystate作为正常参数,但其他所有内容都存储在request.GETQueryDict 中。

于 2011-03-23T08:41:02.840 回答
2

您也可以只创建一个 url(仅检查路径的开头,应该相同)指向您的视图,然后request.path在您的视图中解析。另一方面,如果您在各种组合中确实有很多可选的过滤器参数,最好的解决方案通常是通过GET-parameters 进行过滤,特别是如果用于过滤的 url 不需要针对任何搜索引擎进行优化。 .

于 2011-03-23T00:21:31.040 回答
-1

尝试在你的 urls.py 中使用类似的东西:

url(r'^(?P<city>[^/]+)/(?P<state>[^/]+)/(radius/(?P<miles>[^/]+)/|company/(?P<company_name>[^/]+)/)*$', 'view')
于 2011-03-22T23:25:07.463 回答