
在 Django Admin 中保存模型时,可通过重写 ModelAdmin 的 save_model 方法,将当前登录用户自动赋值给模型字段(如 added_by),从而记录首次创建者。关键在于逻辑应放在 ModelAdmin 类中,而非模型本身,且需使用 obj._state.adding 判断是否为新建对象。
在 django admin 中保存模型时,可通过重写 modeladmin 的 `save_model` 方法,将当前登录用户自动赋值给模型字段(如 `added_by`),从而记录首次创建者。关键在于逻辑应放在 `modeladmin` 类中,而非模型本身,且需使用 `obj._state.adding` 判断是否为新建对象。
要实现在 Django 后台管理界面中自动记录模型的创建者用户,必须将业务逻辑实现在 ModelAdmin 子类中,而不是错误地尝试在模型(models.py)里定义 save_model 方法——因为 save_model 是 Django Admin 的钩子方法,仅在 ModelAdmin 类中有效,模型类中定义该方法不会被调用。
以下是正确、健壮的实现步骤:
✅ 正确实现方式
首先,确保模型中已正确定义外键字段(推荐使用 on_delete=models.SET_NULL 以避免级联删除导致数据异常):
# models.py
from django.conf import settings
from django.db import models
class Project(models.Model):
name = models.CharField(max_length=200)
# 其他字段...
added_by = models.ForeignKey(
settings.AUTH_USER_MODEL,
on_delete=models.SET_NULL, # 更安全:用户删除后保留记录
null=True,
blank=True,
related_name='created_projects'
)然后,在 admin.py 中注册对应的 ModelAdmin,并重写 save_model:
# admin.py
from django.contrib import admin
from .models import Project
@admin.register(Project)
class ProjectAdmin(admin.ModelAdmin):
list_display = ['name', 'added_by', 'created_at']
readonly_fields = ['added_by'] # 可选:防止手动修改
def save_model(self, request, obj, form, change):
# obj._state.adding 为 True 表示这是首次保存(即新增对象)
if not change and not obj.pk: # 等价于 obj._state.adding(更直观)
obj.added_by = request.user
super().save_model(request, obj, form, change)? 为什么用 obj._state.adding 或 not change and not obj.pk?
change 参数由 Django Admin 自动传入,表示本次操作是否为编辑(True)还是新增(False)。而 obj.pk is None 在新建对象未保存前恒为真,二者组合判断比仅依赖 obj.pk 更可靠(尤其在自定义主键或事务边界场景下)。官方文档明确推荐使用 obj._state.adding,它是 Django 内部用于精确标识“是否处于新增流程”的权威标志。
⚠️ 常见误区与注意事项
- ❌ 错误:在 models.py 中定义 save_model 方法 → 不会被 Admin 调用,完全无效;
- ❌ 错误:使用 if not obj.pk: 单独判断 → 在某些特殊场景(如 bulk_create 或预分配主键)下可能失效;
- ✅ 推荐:始终结合 change 参数或直接使用 obj._state.adding;
- ✅ 安全增强:将 added_by 字段设为 readonly_fields,避免管理员在编辑页面意外篡改;
- ✅ 扩展建议:若还需记录最后修改人,可额外添加 updated_by 字段,并在 save_model 中对 change=True 分支赋值。
? 总结
记录创建用户是 Django Admin 开发中的高频需求。核心要点有三:
- 逻辑必须置于 ModelAdmin 子类中;
- 使用 obj._state.adding(或 not change and not obj.pk)精准识别新建动作;
- 模型外键应配置 on_delete=models.SET_NULL 并允许 null=True,兼顾数据完整性与业务可维护性。
遵循此模式,即可稳定、可维护地实现“谁创建,谁留痕”。









