将字段更改为多个时的Django数据迁移

将字段更改为多个时的Django数据迁移,django,django-models,django-migrations,Django,Django Models,Django Migrations,我有一个Django应用程序,我想将一个字段从ForeignKey更改为ManyToManyField。我想保留我的旧数据。对此,最简单/最好的流程是什么?如果重要的话,我使用sqlite3作为我的数据库后端 如果我对这个问题的总结不清楚,下面是一个例子。假设我有两个模型: class Author(models.Model): author = models.CharField(max_length=100) class Book(models.Model): au

我有一个Django应用程序,我想将一个字段从ForeignKey更改为ManyToManyField。我想保留我的旧数据。对此,最简单/最好的流程是什么?如果重要的话,我使用sqlite3作为我的数据库后端

如果我对这个问题的总结不清楚,下面是一个例子。假设我有两个模型:

class Author(models.Model):  
    author = models.CharField(max_length=100) 

class Book(models.Model):  
    author = models.ForeignKey(Author)  
    title = models.CharField(max_length=100)
假设我的数据库中有很多数据。现在,我想更改书本模型,如下所示:

class Book(models.Model):  
    author = models.ManyToManyField(Author)  
    title = models.CharField(max_length=100) 
我不想丢失所有以前的数据

实现这一点的最佳/最简单的方法是什么


Ken

我意识到这个问题由来已久,当时数据迁移的最佳选择是使用South。现在,Django有了自己的migrate命令,流程略有不同

我已经将这些模型添加到一个名为books的应用程序中,如果不是您的情况,请进行相应调整

首先,将字段添加到Book中,并将相关的_名称至少添加到其中一个,或同时添加两个,否则它们将发生冲突:

class Book(models.Model):  
    author = models.ForeignKey(Author, related_name='book')
    authors = models.ManyToManyField(Author, related_name='books')
    title = models.CharField(max_length=100) 
生成迁移:

$ ./manage.py makemigrations
Migrations for 'books':
  0002_auto_20151222_1457.py:
    - Add field authors to book
    - Alter field author on book
现在,创建一个空迁移来保存数据本身的迁移:

./manage.py makemigrations books --empty
    Migrations for 'books':
0003_auto_20151222_1459.py:
并向其中添加以下内容。要确切了解其工作原理,请查看上的文档。注意不要覆盖迁移依赖项

# -*- coding: utf-8 -*-
from __future__ import unicode_literals

from django.db import models, migrations


def make_many_authors(apps, schema_editor):
    """
        Adds the Author object in Book.author to the
        many-to-many relationship in Book.authors
    """
    Book = apps.get_model('books', 'Book')

    for book in Book.objects.all():
        book.authors.add(book.author)


class Migration(migrations.Migration):

    dependencies = [
        ('books', '0002_auto_20151222_1457'),
    ]

    operations = [
        migrations.RunPython(make_many_authors),
    ]
现在从模型中删除author字段-它应该如下所示:

class Book(models.Model):
    authors = models.ManyToManyField(Author, related_name='books')
    title = models.CharField(max_length=100)
为此创建新的迁移,并运行所有迁移:

$ ./manage.py makemigrations
Migrations for 'books':
  0004_remove_book_author.py:
    - Remove field author from book

$ ./manage.py migrate
Operations to perform:
  Synchronize unmigrated apps: messages, staticfiles
  Apply all migrations: admin, auth, sessions, books, contenttypes
Synchronizing apps without migrations:
  Creating tables...
    Running deferred SQL...
  Installing custom SQL...
Running migrations:
  Rendering model states... DONE
  Applying books.0002_auto_20151222_1457... OK
  Applying books.0003_auto_20151222_1459... OK
  Applying books.0004_remove_book_author... OK

就这样。以前可在book.author上找到的作者现在应该位于从book.authors.all获得的查询集中。

您应该做的最好和最简单的事情可能是:

创建具有不同名称的多对多字段,例如

authors = models.ManyToManyField(Author)
编写一个小函数,将foreignkey值转换为M2M值:

def convert():
    books = Book.objects.all()
    for book in books:
        if book.author:
            li = [book.author.id]
            book.authors.append(li)
            book.save()

运行后,您可以从表中删除“作者”字段,然后再次运行迁移。

签出。更具体地说,请签出本教程的“数据迁移”部分:在所有迁移中使用South是一个好习惯。非常好的工具,感谢链接!如果这还不明显,请确保在尝试任何迁移之前备份数据。幸运的是,复制sqlite就像复制cp命令一样简单。我想知道是否有关于这方面的错误报告。嘿@JanSegre:不是真的,多对多关系的更改在您运行时就完成了。add.Oh,您说得对,谢谢。这就是我忽略文档而不是正确阅读文档的原因。有一个建议,如果在一个大表上运行,for循环中的查询会很慢。使用单行SQL会快得多。e、 g.在图书中插入图书作者图书id、作者id选择id、图书作者id;您可以使用migrations.RunSQLyou可以在一次迁移中完成,在操作列表中,在该命令中写入命令:add_m2m_字段,your_function_to_migration_数据,delete_fk_字段,rename_m2m_fieldHI@Salvioner,apps不是导入,这是RunPython传递给许多作者的一个参数。这个答案现在已经5年了,我最近几年没有运行过这段代码,但无论如何,解决这一问题的方法都不是添加导入。在django 1.10中,我得到了一个AttributeError:“ManyRelatedManager”对象没有属性“append”,我删除了。id,并让book.authors=li,这就行了。我相信你也可以做book.authors.addli,这应该可以满足这个问题。你可以随意添加和保存模型。问题和答案更具体地涉及迁移。