尝试在带有条目的子表之后添加父表时,遇到TypeError:__init __()缺少1个必需的位置参数:'on_delete'


90

我的sqlite数据库中有两个类,一个名为的父表,一个名为Categorie的子表Article。我首先创建了子表类并添加了条目。所以首先我有这个:

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

在添加父表之后,现在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'

我在stackoverflow中看到了一些类似的问题,但似乎不是同一个问题:__init __()缺少1个必需的位置参数:'quantity'


3
您正在使用哪种版本的django?
alfonso.kim

1
那么,您对此感到困惑吗?正如错误明确指出的那样,ForeignKey具有必需的参数on_delete。参见文档
Daniel Roseman

我不需要该on_delete参数,它是强制性的吗?
Christian Lisangola

@ jochri3是的,必需的位置参数表示它是必需的。查看文档以找出最适合您需求的选项。
cezar

Answers:


165

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

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

错误应该消失。

最终,您可能需要另一种选择on_delete,请查看文档以了解更多详细信息:

https://docs.djangoproject.com/zh-CN/1.11/ref/models/fields/#django.db.models.ForeignKey

编辑:

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

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

on_delete = models.CASCADE是Django <2
Peter F


11

从Django 2.0开始on_delete是必需的:

用户=模型.OneToOneField(用户,on_delete = models.CASCADE)

如果删除了用户,它将删除子表数据。有关更多详细信息,请参阅Django文档。


1
为什么在Andrey对此信息作出较早响应的同时回答这个问题?
塞缪尔·道祖

11

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

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

这是on_delete中可以使用的一些方法

  1. 级联

级联删除。Django会在DELETE CASCADE上模拟SQL约束的行为,并删除包含ForeignKey的对象

  1. 保护

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

  1. 没做什么

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

您可以通过阅读文档找到有关on_delete的更多信息。


4

如果使用外键,则必须使用“ on_delete = models.CASCADE”,因为它将消除从父表中删除原始元素后产生的复杂性。就如此容易。

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


1

Django 1.9之后的版本on_delete成为必需的参数,即来自Django 2.0。

在旧版本中,它默认为CASCADE。

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

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

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

有关on_delete附带的其他参数的官方文档


0

如果您不知道输入参数的选项。只是想保留默认值,例如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

0

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


-3

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


2
这如何回答这个问题?
cezar '19

嗨,玛雅妮克。您可能做了其他事情来解决该错误,或​​者您正在使用django的早期版本。
克里斯·戴尔

这是通过对“on_delete”参数models.ForeignKey提供值解决
克里斯敢于

您使用的是版本2之前的Django,因为所有更高版本都on_delete具有强制性!在早期版本中,默认情况下是on_delete = models.CASCADE
Optider
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.