python - django 信号是否也包含在 transaction.atomic 装饰器中?

标签 python django django-models transactions django-signals

我有一个模型文件,它使用 post_save 信号在另一个表中创建链接行。以典型的方式,我可以从我的一个 View 创建一个页面,该页面用 @transaction.atomic 装饰。

我想知道这个装饰器是否会将 Page 对象的创建和 SharedPage 对象的创建放在同一个事务中。从 django 文档中不清楚信号是该原子事务的一部分。

模型.py

class Page(models.Model):
    name = models.CharField(default='My default page',max_length=200,blank=False)
    created_at = models.DateTimeField(auto_now_add=True)
    owner = models.ForeignKey(User, on_delete=models.CASCADE)
    slug = models.SlugField()
    uuid = models.UUIDField(default=uuid.uuid4, editable=False)
    is_public = models.BooleanField(default=False)

    def __str__(self):              # __unicode__ on Python 2
        return self.name

    class Meta:
        ordering = ['position','created_at']

@receiver(post_save, sender=Page)
def create_shared_page_entry(sender, instance, created, **kwargs):
    if created:
        shared_page = SharedPage.objects.create(
            page=instance,
            user=instance.user,
            can_edit=True
        )

view.py

@require_http_methods(["POST"])
@transaction.atomic
def page_create(request):
    name = request.POST.get('name')
    page = Page.objects.create(name=name, owner=request.user)

    data = serializers.serialize("json", [page])
    return HttpResponse(data, content_type='application/json')

最佳答案

是的,为连接的自动提交设置分配相同的信号(由 transaction.commit 装饰器调整),用于保存模型。引用 django.db.models.base.Model.save_base() 方法中的代码,

    if not meta.auto_created:
        signals.pre_save.send(sender=origin, instance=self, raw=raw, using=using, update_fields=update_fields)

    with transaction.atomic(using=using, savepoint=False):
        if not raw:
            self._save_parents(cls, using, update_fields)
        updated = self._save_table(raw, cls, force_insert, force_update, using, update_fields)
    # Store the database on which the object was saved
    self._state.db = using
    # Once saved, this is no longer a to-be-added instance.
    self._state.adding = False

    # Signal that the save is complete
    if not meta.auto_created:
        signals.post_save.send(sender=origin, instance=self, created=(not updated),update_fields=update_fields, raw=raw, using=using)

如您所见,没有编写特殊代码来更改自动提交设置。因此,如果您的 View 声明所有与数据库相关的内容都必须使用@transaction.atomic 确保原子性,那么您的 View (model.save() 或通过信号处理程序)所做的数据库更改不会提交,直到您的 View 执行完毕。

希望对您有所帮助。

关于python - django 信号是否也包含在 transaction.atomic 装饰器中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36331753/

相关文章:

从应用程序更改时,django 简单历史记录不起作用

python - django excel xlwt

Python:Plotly:平行坐标不透明度

python - 在 Ubuntu Azure VM 上找到 SQL Server 的 ODBC 驱动程序 13.0

python - from django.contrib.auth.models import User 与将用户foreignKey定义为owner = models.ForeignKey ('auth.User' )

python - django queryset.update(**kwargs) 上的原子事务

python - Django WSGI多线程和连接问题

python - 谷歌应用引擎和云 SQL : Lost connection to MySQL server at 'reading initial communication packet'

python - Django 中的菜单选项卡突出显示问题

Django模型外键查询集选择相关字段