如何使 Django 模型字段从另一个字段派生其值并且不可编辑

How to make Django model field derive its value from another field and be non editable

我有一个用例,在这个用例中,我在 Django 休息框架中创建了一个自定义用户模型,另一个名为 Book 的模型由用户创建。我的模型如下所示:

from django.db import models
from django.contrib.auth.models import (
    AbstractBaseUser,
    BaseUserManager,
    PermissionsMixin)
from django.conf import settings


class UserManager(BaseUserManager):
    def create_user(self, email, password=None, author_pseudonm=None, **extra_fields):
        """Creates and saves a new user """
        if not email:
            raise ValueError('Users must have an email address')
        if not author_pseudonm:
            raise ValueError("User must have an author pseudo name field")
        user = self.model(email=self.normalize_email(email),
                          author_pseudonm=author_pseudonm, **extra_fields)
        user.set_password(password)
        user.save(using=self._db)
        return user

    def create_superuser(self, email, password, author_pseudonm):
        """Creates and saves a new super user"""
        user = self.create_user(
            email=email, password=password, author_pseudonm=author_pseudonm)
        user.is_staff = True
        user.is_superuser = True
        user.save(using=self._db)
        return user


class User(AbstractBaseUser, PermissionsMixin):
    """Custom user model which supports using email instead of username"""
    email = models.EmailField(max_length=255, unique=True)
    name = models.CharField(max_length=255)
    author_pseudonm = models.CharField(max_length=255)
    is_active = models.BooleanField(default=True)
    is_staff = models.BooleanField(default=False)

    objects = UserManager()

    USERNAME_FIELD = 'email'
    REQUIRED_FIELDS = ["author_pseudonm"]


class Book(models.Model):
    """Custom book model"""
    title = models.CharField(max_length=255, unique=True)
    user = models.ForeignKey(
        settings.AUTH_USER_MODEL,
        on_delete=models.CASCADE
    )
    description = models.TextField()
    price = models.IntegerField()
    author = models.CharField(default=user.author_pseudonm)
    is_published = models.BooleanField(default=True)

    def __str__(self):
        return self.title

在书籍模型中,我有一个名为 user 的字段,它基本上是一个外键,将创建的书链接到创建该书的人和另一个名为 author 的字段,它基本上是来自用户模型的一个名为 author_pseudonm 的字段.

我对 rest 框架还很陌生,我想知道这是否是正确的方法。

我想在创建图书时自动填充 author_pseudonm 字段,并且不可编辑。

请建议我如何使用其余框架实现相同的目标。

基本上您不想以这种方式复制您的数据 - 您不需要将 author 存储在您的 Book 模型中,因为它已经存在于您的 User 中模型。每当你需要获取给定书籍的作者笔名时,你应该简单地做

book.user.author_pseudonm

或者如果您想在 Django 管理或序列化程序或查询集过滤器中引用该列,您可以使用 user__author_pseudonm

(注意:你这里好像也有错别字author_pseudonm -> author_pseudonym)

如果我理解正确,你的问题是用户可以随着时间的推移更改他们的笔名,并且当作者更改他们的笔名时,特定书籍的笔名不应该改变。

我这样做的方法是使用自定义 save 函数,在给定 User 外键的情况下,获取 author_pseudonym 值并将其作为 author 字段并保存模型。您可以在 author 字段定义中放置一个 editable=False 标志,以确保将来不会有人不小心编辑它。

以下是伪代码,但我想 Booksave 函数看起来像这样:

def save(self, *args, **kwargs):
    user = self.user
    self.author = user.author_pseudonm
    super(Book, self).save(**kwargs)

郑重声明,DRF 和 vanilla Django 的 models 部分是相同的。无论您是否使用 DRF,您都将以相同的方式实现这一点,因为这只与模型有关。