在Django模板中执行getattr()风格的查找

55 投票
7 回答
28933 浏览
提问于 2025-04-15 11:29

Python的getattr()方法很有用,特别是当你事先不知道某个属性的名字时。

这个功能在模板中也会很方便,但我一直没找到合适的方法来实现。有没有什么内置的标签或者非内置的标签可以用来动态查找属性呢?

7 个回答

2

我最后在相关的模型里添加了一个方法,这个方法在模板中可以像属性一样被访问。

不过,我觉得如果有一个内置的标签可以让你动态查找属性,那就太好了,因为我们在模板中经常会遇到这个问题。

3

我觉得不是这样。不过,写一个自定义模板标签来返回上下文字典中的一个属性其实并不难。如果你只是想返回一个字符串,可以试试下面的这个:

class GetAttrNode(template.Node):
    def __init__(self, attr_name):
        self.attr_name = attr_name

    def render(self, context):
        try:
            return context[self.attr_name]
        except:
            # (better yet, return an exception here)
            return ''

@register.tag
def get_attr(parser, token):
    return GetAttrNode(token)

需要注意的是,直接在你的视图里做这件事可能和在模板里做一样简单,除非这个条件在你的数据中经常出现。

74

我最近也需要写这个代码作为一个自定义模板标签。为了处理所有的查找情况,它首先会进行标准的属性查找,然后尝试进行字典查找,再接着尝试进行getitem查找(这样列表也能正常工作),最后在找不到对象时,它会遵循标准的Django模板行为。

(更新于2009-08-26,现在也能处理列表索引查找)

# app/templatetags/getattribute.py

import re
from django import template
from django.conf import settings

numeric_test = re.compile("^\d+$")
register = template.Library()

def getattribute(value, arg):
    """Gets an attribute of an object dynamically from a string name"""

    if hasattr(value, str(arg)):
        return getattr(value, arg)
    elif hasattr(value, 'has_key') and value.has_key(arg):
        return value[arg]
    elif numeric_test.match(str(arg)) and len(value) > int(arg):
        return value[int(arg)]
    else:
        return settings.TEMPLATE_STRING_IF_INVALID

register.filter('getattribute', getattribute)

模板使用示例:

{% load getattribute %}
{{ object|getattribute:dynamic_string_var }}


撰写回答