Django(21)migrate報錯的解決方案


前言

在講解如何解決migrate報錯原因前,我們先要了解migrate做了什么事情,migrate:將新生成的遷移腳本。映射到數據庫中。創建新的表或者修改表的結構。
 

問題1:migrate怎么判斷哪些遷移腳本需要執行?

它會將代碼中的遷移腳本和數據庫中django_migrations中的遷移腳本進行對比,如果發現數據庫中,沒有這個遷移腳本,那么就會執行這個遷移腳本。
 

問題2:migrate做了什么事情

  1. 將相關的遷移腳本翻譯成SQL語句,在數據庫中執行這個SQL語句。
  2. 如果這個SQL語句執行沒有問題,那么就會將這個遷移腳本的名字記錄到django_migrations中。
     

實戰案例

當我們了解清楚migrate的作用后,我們來看一個案例
首先我們創建一個項目orm_migrations_demo,接着創建2個app應用frontarticle,代碼結構如下圖

接着在front.models.pyarticle.models.py中創建模型

# front.models.py
class Article(models.Model):
    name = models.CharField(max_length=200)

# article.models.py
class FrontUser(models.Model):
    name = models.CharField(max_length=200)

接着在settings.pyINSTALL_APPS中將app注冊

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'front',
    'article',
]

接着我們打開命令行,輸入makemigrations article,再輸入makemigrations front,此時2個app目錄中都會出現遷移文件0001_initial.py,此時數據庫中是沒有表的,因為還沒有執行遷移命令
接着我們執行migrate article,再輸入migrate front,migrate發現數據庫中沒有遷移腳本,那么就會執行剛才生成的2個遷移腳本,將遷移腳本翻譯成SQL語句,然后創建了2張表,執行完成后,會將遷移腳本記錄到django_migrations表中,數據庫中表結構如下:

django_migrations表中內容如下:

接下來我們在article.models.py中添加一個content字段

class Article(models.Model):
    name = models.CharField(max_length=200)
    content = models.CharField(max_length=200, null=True)

然后執行命令makemigrations article,會在項目中生成遷移文件0002_article_content.py,接着執行migrate article,執行遷移腳本,此時數據庫中表django_migrations有3個遷移腳本

 

現在我們來模仿錯誤信息內容,我們將數據庫中django_migrations表中的0002_article_content這行記錄刪除,然后我們來看下0002_article_content的代碼

class Migration(migrations.Migration):

    dependencies = [
        ('article', '0001_initial'),
    ]

    operations = [
        migrations.AddField(
            model_name='article',
            name='content',
            field=models.CharField(max_length=200, null=True),
        ),
    ]

這個遷移腳本的作用是為article模型添加content字段,但是我們現在看一下article中的字段:

從上圖中我們可以清楚的看到article表中已經有了content字段,那么我們再執行migrate article命令時,就會報錯,說content字段重復了,報錯信息如下

django.db.utils.OperationalError: (1060, "Duplicate column name 'content'")

如果發生這種報錯信息,解決辦法是在migrate命名后添加參數--fake--fake可以將指定的遷移腳本名字添加到數據庫中。但是並不會把遷移腳本轉換為SQL語句去修改數據庫中的表
所以,我們可以執行命名migrate article --fake,會在django_migrations表中插入遷移腳本記錄0002_article_content,如下圖

此時數據庫中表結構和django中的表結構完全一致,接下來執行遷移命令,就不會報錯了
 

第一種報錯情況總結

原因:執行migrate命令會報錯的原因是。數據庫的django_migrations表中的遷移版本記錄和代碼中的遷移腳本不一致導致的。
解決辦法:使用--fake參數:首先對比數據庫中的遷移腳本和代碼中的遷移腳本。然后找到哪個不同,之后再使用--fake,將代碼中的遷移腳本添加到django_migrations中,但是並不會執行sql語句。這樣就可以避免每次執行migrate的時候,都執行一些重復的遷移腳本。
 

第二種報錯情況

如果我們不管怎么執行migrate命令都會報錯,那么就執行第二種方案

  1. 將出問題的app下的所有模型,都和數據庫中的表保持一致。
  2. 將出問題的app下的所有遷移腳本文件都刪掉。再在django_migrations表中將出問題的app相關的遷移記錄都刪掉。
  3. 使用makemigrations,重新將模型生成一個遷移腳本。
  4. 使用migrate --fake-initial參數,將剛剛生成的遷移腳本,標記為已經完成(因為這些模型相對應的表,其實都已經在數據庫中存在了,不需要重復執行了。)
  5. 可以做其他的映射了。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM