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

5
具体来说,我有django模型对象,我想在运行时添加属性。如果这适用于Django以外的任何Python类,那将是很棒的。
我有以下模型:
models.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"

当我使用时
person.newtemporaryproperty=property("")

当我尝试给它添加一个值时,会出现“无法设置属性错误”的提示。可能是我使用方法不正确。

编辑

我想做的是查看模型中的每个人是否参加了活动。如果他们参加了活动,就在他们的名字旁边打勾。以下是其他相关文件:

tables.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),

        }

views.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

你想要做什么?对于Django模型实例来说,这几乎肯定是一个不好的主意。 - sytech
1
无论如何,对于属性,您至少提供属性的getter。如果您希望在将新值分配给属性时发生某些事情,则还需要提供setter。如果您只想要一个常规的属性,只需执行obj.attr ='value'即可。 - sytech
我正在进行计算,以确定人员列表中的某个人是否属于特定类别。如果是,我想将一个没有任何数据库原因的新属性设置为值“checked”,如果不是,则将该值设置为“”,这样当我将其传递给django-tables2类时,它将知道该记录需要一个选中的复选框。@sytech - Bigbob556677
那么你可能想要做的是创建一个方法来查询模型管理器上的内容。此外,您可以在Django模型类上创建Python属性,它不会对数据库产生影响。(当然,这意味着您也无法通过此类属性进行查询!)例如,您可以拥有一个名为is_something的属性,以便每个对象都具有该属性,并且可以在视图、模板等中使用。 - sytech
2
如果您能提供完整的模型,我会更新我的答案,并提供一个您可以使用的查询以达到相同的效果。 - sytech
显示剩余5条评论
1个回答

4

因为 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属性。

for person in Person.objects.all():
    if person.is_in_category:
        # do something

这也适用于模板...例如,如果您想制作一个人员表格。
<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

@Philip556677 - 我更新了我的答案,包括可查询的解决方案。 - sytech

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接