【问题标题】:Using other models in Django Manager causes ImportError: partially initialized module (due to circular dependency)在 Django Manager 中使用其他模型会导致 ImportError: partial initialized module (由于循环依赖)
【发布时间】:2021-05-10 05:38:32
【问题描述】:

我的UserManager 中有以下功能,我在我的 django 应用程序中与我的CustomUser 模型一起使用。

from django.contrib.auth.models import BaseUserManager
class UserManager(BaseUserManager):
    ...
    def _create_user(self, email, password, **extra_fields):
        """Create and save a User with the given email and password."""
        if not email:
            raise ValueError('The given email must be set')
        email = self.normalize_email(email)
        user = self.model(email=email, **extra_fields)
        user.set_password(password)
        user.save()
        return user

我需要在用户注册的时候创建一个对应的UserProfile对象,所以我更新了_create函数如下:

from profiles.models import UserProfile
class UserManager(BaseUserManager):
    def _create_user(self, email, password, **extra_fields):
        """Create and save a User with the given email and password."""
        if not email:
            raise ValueError('The given email must be set')
        email = self.normalize_email(email)
        user = self.model(email=email, **extra_fields)
        user.set_password(password)
        user.save()
        # Create the user profile
        UserProfile.objects.create(user=user)
        return user

但这会引发:

ImportError:无法从部分初始化的模块“用户”导入名称“自定义用户” s.models'(很可能是由于循环导入)(../users/models.py)

我的CustomUser定义如下:

class CustomUser(AbstractUser):
    username = None
    email = models.EmailField(unique=True)

    USERNAME_FIELD = 'email'
    REQUIRED_FIELDS = []
    auth_provider = models.CharField(
        max_length=10,
        default=AUTH_PROVIDERS.get('email'))
    objects = UserManager()

UserProfile模型定义如下:

class UserProfile(models.Model):
    user = models.OneToOneField(
        CustomUser, 
        null=True, 
        on_delete=models.CASCADE, 
        related_name="profile")
    ...
    description = models.CharField(max_length=200, blank=True)

为什么使用UserProfile在管理器中会导致循环导入错误?我明白我是否在CustomUser 模型本身中使用了UserProfile,但管理器不只是数据库查询的接口吗?

【问题讨论】:

    标签: python django django-models


    【解决方案1】:

    来自doc

    这种引用称为惰性关系在解决两个应用程序之间的循环导入依赖关系时很有用 em>。

    因此,您可以使用 app_name.ModelName 语法将 CustomUser 连接到 UserProfile作为

    class UserProfile(models.Model):
        user = models.OneToOneField(
            "my_app.CustomUser" # Change is here, 
            null=True, 
            on_delete=models.CASCADE, 
            related_name="profile")
        ...
        description = models.CharField(max_length=200, blank=True)

    【讨论】:

    • 不幸的是,这导致我的 UserProfile 表单抛出:Cannot create form field for 'user' yet, because its related model '<my_app>.CustomUser' has not been loaded yet
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-12-30
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2023-02-13
    相关资源
    最近更新 更多