首先说明save(commit=False)和save_m2m()都是form表单的方法。
django中有时候需要使用form.save(commit=False)方法,更新对象属性,但并不向数据库真正提交数据,举个博客的例子吧,登录之后,写博客,然后保存博客的内容。博客的有些内容是在界面上由用户填写的,有些内容是在后台添加的。例如,用户姓名之类的。
这样的话就需要在form.save()方法中传递一个参数commit,赋值为False,代表不要提交到数据库。然后给blog.author=request.user赋值,然后有两种选择:
1. 调用new_blog.save()保存blog的基本信息,然后在调用form.save_m2m()保存关联信息(外键字段)。
2. 再一次调用form.save()保存blog基本信息以及关联信息。
django-xadmin中的实例:
edit.py源码:
class ModelFormAdminView(ModelAdminView):
...
@filter_hook
def valid_forms(self):
return self.form_obj.is_valid()
@filter_hook
def save_forms(self):
self.new_obj = self.form_obj.save(commit=False)
@filter_hook
def save_models(self):
self.new_obj.save()
flag = self.org_obj is None and 'create' or 'change'
self.log(flag, self.change_message(), self.new_obj)
@filter_hook
def save_related(self):
self.form_obj.save_m2m()
@csrf_protect_m
@transaction.atomic
@filter_hook
def post(self, request, *args, **kwargs):
self.instance_forms()
self.setup_forms()
if self.valid_forms():
self.save_forms()
self.save_models()
self.save_related()
response = self.post_response()
cls_str = str if six.PY3 else basestring
if isinstance(response, cls_str):
return HttpResponseRedirect(response)
else:
return response
return self.get_response()
重写save_related保存关联字段:
def save_related(self):
super().save_related()
self.new_obj.author_id = 2
self.new_obj.save()
注意一点:在save_models()方法中更新外键字段是无效的,因为更新外键最终需要调用form.save_m2m(),调用form.save_m2m()方法之前对外键字段无论做出如何修改,最终的外键值都是取form表单对象中的的值。
并且更新外键必须放到save()方法后面,否则报needs to have a value for field ”id“ before this many-to-many relationship can be used错误。
@filter_hook
def save_models(self):
self.new_obj.save()
# self.new_obj.author.add(*authors_obj) 无效的操作!
flag = self.org_obj is None and 'create' or 'change'
self.log(flag, self.change_message(), self.new_obj)