如果您事先不知道某个属性的名称,Python的getattr()方法很有用。
此功能在模板中也会派上用场,但我从来没有想过要做到这一点的方法。是否有可以执行动态属性查找的内置标记或非内置标记?
答案 0 :(得分:57)
我最近还必须将此代码编写为自定义模板标记。要处理所有查找方案,它首先执行标准属性查找,然后尝试进行字典查找,然后尝试 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 }}
答案 1 :(得分:2)
我不这么认为。但是编写一个custom template tag来返回上下文字典中的属性并不难。如果您只是想尝试返回一个字符串,请尝试以下方法:
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)
请注意,在您的视图中而不是在模板中执行此操作可能同样容易,除非这是您的数据中经常重复的条件。
答案 2 :(得分:2)
我最终向相关模型添加了一个方法,并且可以像模板中的属性一样访问该方法。
不过,我认为如果内置标记允许您动态查找属性会很好,因为这是我们很多人在模板中经常遇到的问题。
答案 3 :(得分: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
答案 4 :(得分:0)
没有内置标记,但write your own不应该太难。
答案 5 :(得分: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)