在运行时向Django模型添加属性?

2024-04-26 19:01:49 发布

您现在位置:Python中文网/ 问答频道 /正文

具体来说,我有django模型对象,我想在运行时添加属性。如果这能在Django之外的任何python类上运行,那就太好了。在

我有以下型号

在模型.py在

class person(models.Model):
    # ...
    firstname=models.TextField(max_length=100)
    lastname=models.TextField(max_length=100)
    type=models.TextField(max_length=100)
    address = models.ForeignKey(address, null=True)

class event(models.Model):
    date=models.DateTimeField(auto_now=True )
    name=models.CharField(max_length=100)     
    attendees = models.ManyToManyField(person)

def main():
    p = person()
    p.fisrtsname="Firsty"
    p.lastname="Lasty"
    p.addnewproperty(newtemporaryproperty)
    p.newtemporaryproperty="This is a new temporary property"

当我使用

^{pr2}$

当我试图添加一个值时,我得到一个“无法设置属性错误”。 我可能用错了。在

编辑

我想做的是看看模特中的每个人是否都参加过活动。如果他们有,在他们的名字旁边打上一个复选框。以下是其他相关文件

在表格.py在

class AttendeesTable(tables.Table):
    firstname = tables.Column()
    lastname = tables.Column()
    here = tables.TemplateColumn('<input id="attendee_{{ record.pk }}" {{ 
record.here }} type="checkbox" />',
                             verbose_name="Here?")

    class Meta:
        attrs = {'id': 'attendancetable', 'width': '100%', 'class': 'table 
table-hover'}
        template = 'django_tables2/bootstrap.html'
        row_attrs = {
            'id': lambda record: str(record.pk),

        }

在视图.py在

def markattendancepage(request):
    person.here = ""
    people= person.objects.all()
    groups = group.objects.all()
    eventid= 1 #set to 1 for testing
    table=None
    for p in people:

        if event.objects.filter(id=eventid, attendees__pk=p.id).exists():

            p.here = "checked"
        else:
            p.here = ""

        table = app.tables.AttendeesTable(people)


    RequestConfig(request, paginate=False).configure(table)
    return render(request, 'app/user/attendancechecker.html', {'attendeetable':table})

    pass

Tags: pyidtablesheremodelsrequesttablerecord
1条回答
网友
1楼 · 发布于 2024-04-26 19:01:49

因为django模型实例经常在后台重新加载,所以您可能不希望在运行时设置属性,因为它们很容易丢失。相反,您可能希望在类定义中有一个属性(或方法),例如

class Person(models.Model):
    first_name=models.CharField(max_length=100)
    last_name=models.CharField(max_length=100)

    @property
    def is_in_category(self):
        # 'calculation' return a boolean
        return True if something else False

如果您有Person的任何特定实例,您可以检查is_in_category属性

^{pr2}$

这也适用于模板。。。例如,如果你想做一张桌子

<table><tr><th>Name</th><th>In category?</th></tr>
{% for person in people %}
    <tr>
      <td>{{ person.first_name }}, {{person.last_name}}</td>
      <td>{% if person.is_in_category %}Yes{% else %}No{% endif %}</td>
    </tr>
{% endfor %}
</table>

但是,由于属性仅作为Python构造存在,因此不能使用基于此属性的SQL查询。在

# this will not work
people_in_category = Person.objects.filter(is_in_category=False)

如果您想执行这样的查询,您需要在模型或相关模型上创建一个字段,或者使用一个与属性等效的SQL表达式。在

编辑:

对于您的模型,您可以执行应该执行相同操作的查询。您的event模型有一个attendees字段,这将是您要查找的,并且是实现此操作的方法,因为它显示您手头有事件id并可以访问事件模型。在

evnt = event.objects.get(pk=eventid)
evnt.attendees # people who attended the event
did_not_attend = people.objects.exclude(id__in=evnt.attendees)

您可能需要考虑在模型管理器上创建一个方法,该方法为查询添加注释,以获得与属性所给您的效果相同的效果。例如

class PersonManager(models.Manager):
    def for_event(self, evnt):
        attended_event = person.objects.filter(id__in=evnt.attendees)
        qs = self.get_queryset()
        return qs.annotate(attended=Exists(attended_event))

如果你用你的员工模型注册这个经理

for p in person.objects.for_event(evnt):
    if p.attended:
        # they attended the event

相关问题 更多 >