比较Django模型中的旧字段和更新字段

1 投票
3 回答
2016 浏览
提问于 2025-04-17 08:02

我想比较模型中旧的和更新的字段。我已经为一个字段解决了这个问题,但我想对所有字段都这样做:

class MyUser(User)
    def save(self, **kwargs):
       if self.pk is not None:
         orig = MyUser.objects.get(pk=self.pk)
         orig_field_names = orig._meta.get_all_field_names()
         field_names = self._meta.get_all_field_names()
         # I want do this in loop 
         if orig.first_name != self.first_name:
           print 'first_name changed'
           UpdateLog.objects.create(
                user = orig,
                filed_name = self.first_name,
                update_time = datetime.now()
             )
         super(MyUser, self).save(**kwargs)

提前谢谢你们!

3 个回答

0

在投票之前先看看文档 -1。捕捉信号是做这件事的更好方法。

1

你需要一个信号。下面是那个链接中介绍的内容,简单来说就是:

Django有一个“信号分发器”,它可以帮助不同的应用程序在框架中发生某些操作时得到通知。简单来说,信号允许某些发送者通知一组接收者,某个动作已经发生。当很多代码都对同一事件感兴趣时,信号特别有用。

Django提供了一些内置的信号,让用户的代码可以在某些操作发生时得到Django本身的通知。

3

这是我用来比较字段的常用函数。处理外键的时候会有点复杂,但总体来说还算不错:

def get_changes_between_objects(object1, object2, excludes=[]):
    """
    Finds the changes between the common fields on two objects

    :param object1: The first object
    :param object2: The second object
    :param excludes: A list of field names to exclude
    """
    changes = {}

    # For every field in the model
    for field in object1._meta.fields:
        # Don't process excluded fields or automatically updating fields
        if not field.name in excludes and not isinstance(field, fields.AutoField):
            # If the field isn't a related field (i.e. a foreign key)..
            if not isinstance(field, fields.related.RelatedField):
                old_val = field.value_from_object(object1)
                new_val = field.value_from_object(object2)
                # If the old value doesn't equal the new value, and they're
                # not both equivalent to null (i.e. None and "")
                if old_val != new_val and not(not old_val and not new_val):
                    changes[field.verbose_name] = (old_val, new_val)

            # If the field is a related field..
            elif isinstance(field, fields.related.RelatedField):
                if field.value_from_object(object1) != field.value_from_object(object2):
                    old_pk = field.value_from_object(object1)
                    try:
                        old_val = field.related.parent_model.objects.get(pk=old_pk)
                    except field.related.parent_model.DoesNotExist:
                        old_val = None

                    new_pk = field.value_from_object(object2)
                    try:
                        new_val = field.related.parent_model.objects.get(pk=new_pk)
                    except field.related.parent_model.DoesNotExist:
                        new_val = None

                    changes[field.verbose_name] = (old_val, new_val)

    return changes

用法:

>>> item = Item.objects.get(pk=1)
>>> item_old = Item.objects.get(pk=1)
>>> print item.my_attribute
'foo'
>>> item.my_attribute = 'bar'
>>> get_changes_between_objects(item, item_old)
{'My Attribute': ('bar', 'foo')}

撰写回答