在 django 模板中执行 getattr() 样式查找

2025-03-05 09:18:00
admin
原创
75
摘要:问题描述:getattr()当你事先不知道某个属性的名称时,Python 的方法很有用。此功能在模板中也很有用,但我从未找到实现它的方法。是否有可以执行动态属性查找的内置标签或非内置标签?解决方案 1:最近我还必须将此代码编写为自定义模板标记。为了处理所有查找场景,它首先进行标准属性查找,然后尝试进行字典查找...

问题描述:

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

此功能在模板中也很有用,但我从未找到实现它的方法。是否有可以执行动态属性查找的内置标签或非内置标签?


解决方案 1:

最近我还必须将此代码编写为自定义模板标记。为了处理所有查找场景,它首先进行标准属性查找,然后尝试进行字典查找,然后尝试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 }}

解决方案 2:

我不这么认为。但编写自定义模板标签来返回上下文字典中的属性并不太难。如果您只是想返回一个字符串,请尝试以下方法:

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)

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

解决方案 3:

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

不过,我认为如果内置标签允许您动态查找属性那就太好了,因为这是我们很多人在模板中经常遇到的问题。

解决方案 4:

保持 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

解决方案 5:

这段代码拯救了我,但我需要它来跨越关系,所以我将其改为用“.”分割参数并递归获取值。它可以在一行中完成:
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)

解决方案 6:

没有内置标签,但编写自己的标签应该不太难。

解决方案 7:

我已经基于@fotinakis 解决方案开发了一个更通用的解决方案,它用于查找字符串表达式的值(无论它是属性还是函数),并且还支持对象链接。

import re
import types

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

def get_attr(object, arg):
    if hasattr(object, str(arg)):
        attr = getattr(object, arg)
        if type(getattr(object, str(arg))) == types.MethodType:
            return attr()
        return attr
    elif hasattr(object, 'has_key') and object.has_key(arg):
        return object[arg]
    elif numeric_test.match(str(arg)) and len(object) > int(arg):
        return object[int(arg)]
    else:
        return object


@register.simple_tag(takes_context=True)
def get_by_name(context, name):
    """"Get variable by string name {% get_by_name data_name.data_func... %}"""
    print(context['instance'].get_edit_url())
    arr = name.split('.')
    obj = arr[0]
    object = context[obj]
    if len(arr) > 1:
        for ar in arr:
            object = get_attr(object, ar)
    return object
相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   2560  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1552  
  IPD(Integrated Product Development)流程作为一种先进的产品开发管理模式,在众多企业中得到了广泛应用。其中,技术评审与决策评审是IPD流程中至关重要的环节,它们既有明显的区别,又存在紧密的协同关系。深入理解这两者的区别与协同,对于企业有效实施IPD流程,提升产品开发效率与质量具有重要意义...
IPD管理流程   1  
  本文介绍了以下10款项目管理软件工具:禅道项目管理软件、ClickUp、Freshdesk、GanttPRO、Planview、Smartsheet、Asana、Nifty、HubPlanner、Teamwork。在当今快速变化的商业环境中,项目管理软件已成为企业提升效率、优化资源分配和确保项目按时交付的关键工具。然而...
项目管理系统   2  
  建设工程项目质量关乎社会公众的生命财产安全,也影响着企业的声誉和可持续发展。高质量的建设工程不仅能为使用者提供舒适、安全的环境,还能提升城市形象,推动经济的健康发展。在实际的项目操作中,诸多因素会对工程质量产生影响,从规划设计到施工建设,再到后期的验收维护,每一个环节都至关重要。因此,探寻并运用有效的方法来提升建设工程...
工程项目管理制度   3  
热门文章
项目管理软件有哪些?
曾咪二维码

扫码咨询,免费领取项目管理大礼包!

云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用