Getting TypeError: __init__() missing 1 required positional argument: 'on_delete' when trying to add parent table after child table with entries

Getting TypeError: __init__() missing 1 required positional argument: 'on_delete' when trying to add parent table after child table with entries

我的 sqlite 数据库中有两个 class,一个 parent table 名为 Categorie 和 child table 名为 Article。我首先创建了 child table class 并添加了条目。所以首先我有这个:

class Article(models.Model):
    titre=models.CharField(max_length=100)
    auteur=models.CharField(max_length=42)
    contenu=models.TextField(null=True)
    date=models.DateTimeField(
        auto_now_add=True,
        auto_now=False,
        verbose_name="Date de parution"
    )

    def __str__(self):
        return self.titre

在我添加 parent table 之后,现在我的 models.py 看起来像这样:

from django.db import models

# Create your models here.
class Categorie(models.Model):
    nom = models.CharField(max_length=30)

    def __str__(self):
        return self.nom


class Article(models.Model):
    titre=models.CharField(max_length=100)
    auteur=models.CharField(max_length=42)
    contenu=models.TextField(null=True)
    date=models.DateTimeField(
        auto_now_add=True,
        auto_now=False,
        verbose_name="Date de parution"
    )
    categorie = models.ForeignKey('Categorie')

    def __str__(self):
        return self.titre

所以当我 运行 python manage.py makemigrations <my_app_name> 时,我得到这个错误:

Traceback (most recent call last):
  File "manage.py", line 15, in <module>
    execute_from_command_line(sys.argv)
  File "C:\Users\lislis\AppData\Local\Programs\Python\Python35-32\lib\site-packages\django-2.0-py3.5.egg\django\core\management\__init__.py", line 354, in execute_from_command_line
    utility.execute()
  File "C:\Users\lislis\AppData\Local\Programs\Python\Python35-32\lib\site-packages\django-2.0-py3.5.egg\django\core\management\__init__.py", line 330, in execute
    django.setup()
  File "C:\Users\lislis\AppData\Local\Programs\Python\Python35-32\lib\site-packages\django-2.0-py3.5.egg\django\__init__.py", line 24, in setup
    apps.populate(settings.INSTALLED_APPS)
  File "C:\Users\lislis\AppData\Local\Programs\Python\Python35-32\lib\site-packages\django-2.0-py3.5.egg\django\apps\registry.py", line 112, in populate
    app_config.import_models()
  File "C:\Users\lislis\AppData\Local\Programs\Python\Python35-32\lib\site-packages\django-2.0-py3.5.egg\django\apps\config.py", line 198, in import_models
    self.models_module = import_module(models_module_name)
  File "C:\Users\lislis\AppData\Local\Programs\Python\Python35-32\lib\importlib\__init__.py", line 126, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
  File "<frozen importlib._bootstrap>", line 986, in _gcd_import
  File "<frozen importlib._bootstrap>", line 969, in _find_and_load
  File "<frozen importlib._bootstrap>", line 958, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 673, in _load_unlocked
  File "<frozen importlib._bootstrap_external>", line 665, in exec_module
  File "<frozen importlib._bootstrap>", line 222, in _call_with_frames_removed
  File "C:\Users\lislis\Django\mon_site\blog\models.py", line 6, in <module>
    class Article(models.Model):
  File "C:\Users\lislis\Django\mon_site\blog\models.py", line 16, in Article
    categorie = models.ForeignKey('Categorie')
TypeError: __init__() missing 1 required positional argument: 'on_delete'

我在 Whosebug 中看到了一些类似的问题,但似乎不是同一个问题:

您可以像这样更改 class Article 的 属性 categorie

categorie = models.ForeignKey(
    'Categorie',
    on_delete=models.CASCADE,
)

错误应该会消失。

最终您可能需要 on_delete 的另一个选项,请查看文档以获取更多详细信息:

https://docs.djangoproject.com/en/1.11/ref/models/fields/#django.db.models.ForeignKey

编辑:

正如您在评论中所述,您对 on_delete 没有任何特殊要求,您可以使用选项 DO_NOTHING:

# ...
on_delete=models.DO_NOTHING,
# ...

由于 Django 2.x,on_delete 是必需的。

Django Documentation

Deprecated since version 1.9: on_delete will become a required argument in Django 2.0. In older versions it defaults to CASCADE.

从 Django 2.0 on_delete 需要:

user = models.OneToOneField(User, on_delete=models.CASCADE)

如果用户被删除,子table数据也会被删除。有关详细信息,请查看 Django 文档。

如果您正在使用外键,那么您必须使用 "on_delete=models.CASCADE",因为它将消除从父 table 中删除原始元素后产生的复杂性。就这么简单。

categorie = models.ForeignKey('Categorie', on_delete=models.CASCADE)

如果对on_delete

的任何人有帮助,以下是可用的选项

CASCADE, DO_NOTHING, PROTECT, SET, SET_DEFAULT, SET_NULL

从 Django 2.0 开始,ForeignKey 字段需要两个位置参数:

  1. 要映射到的模型
  2. on_delete 参数
categorie = models.ForeignKey('Categorie', on_delete=models.PROTECT)

这里有一些方法可以用在on_delete

  1. 级联

级联删除。 Django 模拟 SQL 约束的行为 ON DELETE CASCADE 并且还删除包含 ForeignKey

的对象
  1. 保护

通过引发 django.db.IntegrityError 的子类 ProtectedError 来防止删除引用的对象。

  1. DO_NOTHING

不采取任何行动。如果您的数据库后端强制执行参照完整性,这将导致 IntegrityError,除非您手动向数据库字段添加 SQL ON DELETE 约束。

您可以通过阅读 documentation 找到更多关于 on_delete 的信息。

这对我有用 pip install django-csvimport --upgrade

如果您不知道输入参数的选项。 只想在迁移前保留默认值,如 on_delete=None

on_delete=models.CASCADE

这是旧版本中的代码片段:

if on_delete is None:
    warnings.warn(
        "on_delete will be a required arg for %s in Django 2.0. Set "
        "it to models.CASCADE on models and in existing migrations "
        "if you want to maintain the current default behavior. "
        "See https://docs.djangoproject.com/en/%s/ref/models/fields/"
        "#django.db.models.ForeignKey.on_delete" % (
            self.__class__.__name__,
            get_docs_version(),
        ),
        RemovedInDjango20Warning, 2)
    on_delete = CASCADE

有一个类似的问题,通过将这两个参数都添加到 ForeignKey 来解决: null=真,on_delete=models.SET_NULL

Post Django version 1.9, on_delete became a required argument, i.e. from Django 2.0.

In older versions, it defaults to CASCADE.

因此,如果您想复制您在早期版本中使用的功能。使用以下参数。

categorie = models.ForeignKey('Categorie', on_delete = models.CASCADE)

这将具有与早期版本相同的效果,无需明确指定。

Official Documentation on other arguments that go with on_delete

对我来说,包管理器正在将 djangorestframework 解析为较低版本,而 authtoken 模型没有 on_delete。我所有的模型都很好。您可以 运行 django.apps.apps.get_models() 获取所有模型的列表,以查看哪些库创建了自己的模型,这可能是问题所在。

"models.ForeignKey()""models.OneToOneField" 必须有 " on_delete"Django 2.0。例如下面:

"models.ForeignKey()":

categories = models.ForeignKey('Category', on_delete=models.CASCADE)

"models.OneToOneField":

categories = models.OneToOneField('Category', on_delete=models.PROTECT)