50

当您事先不知道某个属性的名称时,Python 的 getattr() 方法很有用。

这个功能也可以在模板中派上用场,但我从来没有想出办法做到这一点。是否有可以执行动态属性查找的内置标签或非内置标签?

4

6 回答 6

67

我最近还不得不将此代码编写为自定义模板标签。为了处理所有查找场景,它首先执行标准属性查找,然后尝试执行字典查找,然后尝试getitem查找(以使列表工作),然后在对象被查找时遵循标准 Django 模板行为未找到。

(2009 年 8 月 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 }}


于 2009-07-10T22:09:44.203 回答
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)

请注意,在您的视图中而不是在模板中执行此操作可能同样容易,除非这是在您的数据中经常重复的条件。

于 2009-05-10T05:35:31.047 回答
2

我最终向有问题的模型添加了一个方法,并且可以像模板中的属性一样访问该方法。

尽管如此,我认为如果内置标签允许您动态查找属性会很棒,因为这是我们很多人在模板中经常遇到的问题。

于 2009-05-10T06:32:43.590 回答
2

保持 get 和 getattr 之间的区别,

@register.filter(name='get')
def get(o, index):
    try:
        return o[index]
    except:
        return settings.TEMPLATE_STRING_IF_INVALID


@register.filter(name='getattr')
def getattrfilter(o, attr):
    try:
        return getattr(o, attr)
    except:
        return settings.TEMPLATE_STRING_IF_INVALID
于 2012-03-19T22:01:55.890 回答
0

没有内置标签,但编写自己的.

于 2009-05-10T05:29:51.060 回答
0

该片段节省了我的时间,但我需要它来跨越关系,所以我将其更改为将 arg 拆分为“。” 并递归获取值。它可以在一行中完成: return getattribute(getattribute(value,str(arg).split(".")[0]),".".join(str(arg).split(".")[1:])) 但为了便于阅读,我将其保留为 4。我希望有人对此有用。

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 AND recursively from a string name"""
    if "." in str(arg):
        firstarg = str(arg).split(".")[0]
        value = getattribute(value,firstarg)
        arg = ".".join(str(arg).split(".")[1:])
        return getattribute(value,arg)
    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
        return 'no attr.' + str(arg) + 'for:' + str(value)

register.filter('getattribute', getattribute)
于 2015-01-15T13:03:27.023 回答