获取 TypeError:__init__() 缺少 1 个必需的位置参数:尝试在包含条目的子表之后添加父表时出现“on_delete”
Posted
技术标签:
【中文标题】获取 TypeError:__init__() 缺少 1 个必需的位置参数:尝试在包含条目的子表之后添加父表时出现“on_delete”【英文标题】:Getting TypeError: __init__() missing 1 required positional argument: 'on_delete' when trying to add parent table after child table with entries 【发布时间】:2017-10-17 00:23:23 【问题描述】:我的 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'
我在***中看到过一些类似的问题,但似乎不是同一个问题:__init__() missing 1 required positional argument: 'quantity'
【问题讨论】:
你用的是哪个版本的 django? 那么你对这里有什么困惑?正如错误清楚地表明,ForeignKey 有一个必需的参数,on_delete
。见the docs。
我不需要on_delete
参数,它是强制性的吗?
@jochri3 是的,必需的位置参数意味着它是强制性的。查看文档以找出最适合您需求的选项。
【参考方案1】:
您可以像这样更改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,
# ...
【讨论】:
on_delete=models.CASCADE 是 Django 中的默认值 【参考方案2】:从 Django 2.x 开始,on_delete
是必需的。
Django Documentation
自 1.9 版起已弃用:on_delete 将成为必需项 Django 2.0 中的参数。在旧版本中,它默认为 CASCADE。
【讨论】:
【参考方案3】:从 Django 2.0 开始,ForeignKey 字段需要两个位置参数:
-
要映射到的模型
on_delete 参数
categorie = models.ForeignKey('Categorie', on_delete=models.PROTECT)
这里有一些可以在 on_delete 中使用的方法
-
级联
级联删除。 Django 模拟 SQL 约束 ON DELETE CASCADE 的行为,并删除包含 ForeignKey 的对象
-
保护
通过引发 django.db.IntegrityError 的子类 ProtectedError 来防止删除引用的对象。
-
什么都不做
不采取任何行动。如果您的数据库后端强制执行引用完整性,这将导致 IntegrityError,除非您手动将 SQL ON DELETE 约束添加到数据库字段。
您可以通过阅读documentation 找到有关 on_delete 的更多信息。
【讨论】:
【参考方案4】:来自 Django 2.0 on_delete
是必需的:
user = models.OneToOneField(User, on_delete=models.CASCADE)
如果用户被删除,它会删除子表数据。有关更多详细信息,请查看 Django 文档。
【讨论】:
为什么这个答案,而安德烈早些时候用这个信息回应?【参考方案5】:如果您使用外键,则必须使用“on_delete=models.CASCADE”,因为它将消除从父表中删除原始元素后产生的复杂性。就这么简单。
categorie = models.ForeignKey('Categorie', on_delete=models.CASCADE)
【讨论】:
【参考方案6】:如果对 on_delete 有帮助,这里有一些可用的选项
CASCADE、DO_NOTHING、PROTECT、SET、SET_DEFAULT、SET_NULL
【讨论】:
【参考方案7】:发布 Django 版本 1.9,
on_delete
成为必需参数,即来自 Django 2.0。
在旧版本中,默认为 CASCADE。
因此,如果您想复制您在早期版本中使用的功能。使用以下参数。
categorie = models.ForeignKey('Categorie', on_delete = models.CASCADE)
这将具有与早期版本相同的效果,无需明确指定。
Official Documentation on other arguments that go with on_delete
【讨论】:
【参考方案8】:如果您不知道输入参数的选项。
只想在迁移前保留on_delete=None
之类的默认值:
on_delete=models.CASCADE
这是旧版本的代码sn-p:
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
【讨论】:
【参考方案9】:通过将这两个参数添加到 ForeignKey 解决了类似的问题: null=True, on_delete=models.SET_NULL
【讨论】:
【参考方案10】:这对我有用
pip install django-csvimport --upgrade
【讨论】:
这如何回答这个问题? 嗨,Mayank。您可能采取了其他措施来修复该错误,或者您使用的是早期版本的 django。 这可以通过为 models.ForeignKey 提供“on_delete”参数的值来解决 您使用的是版本 2 之前的 Django。因为所有以后的版本都将on_delete
强制要求!在早期版本中默认为on_delete = models.CASCADE
以上是关于获取 TypeError:__init__() 缺少 1 个必需的位置参数:尝试在包含条目的子表之后添加父表时出现“on_delete”的主要内容,如果未能解决你的问题,请参考以下文章
TypeError: __init__() 得到了一个意外的关键字参数“评分”
TypeError: __init__() 接受 2 个位置参数,但给出了 4 个
TypeError: __init__() 接受 1 个位置参数,但给出了 3 个
TypeError: __init__() 接受 1 个位置参数,但给出了 2 个