Django笔记二十之手动编写migration文件

本文首发于公众号:Hunter后端
原文链接:Django笔记二十之手动编写migration文件

前面介绍过,migration 文件主要记录的是 Django 系统 model 的变化,然后通过 migrate 命令将变化适配到数据库中。

比如在某个 application 下新增了某张表,或者对某张表更改了字段,可以生成 migration 文件,然后通过 migrate 更改到数据库。

除了系统能够自动生成的,我们还可以手动创建 migration 文件来操作数据库,这个用途主要是用于比如,创建表后,需要写入一些初始化的数据的情况。

  1. 基础命令
  2. migration文件介绍
  3. 自定义migration文件
  4. RunSQL()
  5. RunPython()

1、基础命令

关于 migration 的命令有如下几条:

  • makemigrations
  • migrate
  • sqlmigrate
  • showmigrations

其中 前面三条命令在第二篇笔记中已经介绍过使用方法,这里介绍一下 showmigrations。

这个作用主要是查看某个 application 下的migration 文件是否已经被更改到数据库中,可以在 Django 系统的根目录用下面的命令测试:

python3 manage.py showmigrations blog

可以看到下面的输出:

blog
 [X] 0001_initial
 [X] 0002_auto_20220118_0926
 [X] 0003_auto_20220121_1016

其中,前面的 [X] 表示已经被更改到数据库中,如果我们再对 blog 的 model 进行任意修改,然后执行 makemigrations 的操作,再次执行 showmigrations 的操作,可以看到下面的输出:

blog
 [X] 0001_initial
 [X] 0002_auto_20220118_0926
 [X] 0003_auto_20220121_1016
 [ ] 0004_alter_book_price

可以看到最下面的一条记录 [] 中是没有 X 的,表示这条 migration 文件没有被执行 migrate。

2、migration文件介绍

每一次通过 makemigrations 生成的 migration 文件都存在系统中,一个最基础的 migration 文件像下面这样:

from django.db import migrations, models


class Migration(migrations.Migration):


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


    operations = [
        migrations.DeleteModel('Tribble'),
        migrations.AddField('Author', 'rating', models.IntegerField(default=0)),
    ]

一个 Migration 的类下,有两个参数,一个是 dependencies,一个是 operations

dependencies 作用是定位上一个执行的 migration 文件的地方,因为每一次 migrate 的执行都是按照顺序的

且他的参数是一个列表,列表的元素是一个元组,里面有两个参数,一个是 application 的名称,一个是上一次运行的 migration 文件,他是可以指定到多个 application 的,意义为在某两个 application 的 migration 文件之后再执行

operations 的作用是 migration 里需要执行的操作,可以是字段的增加、删除、修改、也可以是表的创建和删除

一个 migration 在执行 migrate 前,我们可以手动对其修改,甚至可以完全自己来定义

3、自定义migration文件

前面介绍了 migration 文件的基本结构,其中有一些关于字段和 model 的操作方法,这些操作都可以通过 makemigration 的方式自动生成。

我们自定义的 migration 文件,与上面的保持一致即可,自定义的 migration 文件需要修改的地方是 operations 里的元素。

假设我们有这样一个需求,创建一张基础映射表后,里面是系统运行所必需的数据,需要在创建表后立即写入,那么就用到了我们这个自定义的 migration 文件。

除了对表字段或者表的修改,还有两种方法实现数据的写入,

一种是使用 SQL 语句插入,用到的migration的函数是 RunSQL()

一种是使用 Django 的 ORM 语句,写 python 的函数来插入,函数是 RunPython

假设创建 Blog 表的migration file 是 0001_create_blog.py

现在需要对其插入两条数据,name 和 tagline 分别是 ('name_1', 'tagline_1') 和 ('name_2', 'tagline_2')

下面用 RunSQL() 和 RunPython() 两种方式来分别介绍。

4、RunSQL()

RunSQL() 函数接受一个字符串,或者一个数组作为参数,参数的内容都是 SQL 语句,这也是为什么函数名为 RunSQL()。

字符串的形式为完整的 SQL 语句,比如我们需要插入这两条数据,则是:

migrations.RunSQL(
	"INSERT INTO blog_blog (name, tagline) values('name_x_4', 'tagline_1'), ('name_x_5', 'tagline_2');"
)

如果是作为数组传入,形式则是:

migrations.RunSQL(
    sql=[
        (
            "INSERT INTO blog_blog (name, tagline) values(%s, %s), (%s, %s);",
            ['name_x_6', 'tagline_1', 'name_x_7', 'tagline_2']
        )
    ]
)

在数组的传入形式中,我们将需要插入的数据都放到一个数组中传入

reverse_sql
RunSQL() 函数除了 sql 参数,还有一个 reverse_sql 参数,用途是 sql 参数执行的 SQL 语句没有执行成功的情况下的一种操作,一般是用于防止数据污染。

假设说我们的 sql 为插入数据,但是因为某种原因,这条语句没有正确插入,报错了,那么系统就会执行 reverse_sql 中的语句,作为一个可逆的操作。

以下是官方的一个使用示例:

migrations.RunSQL(
    sql=[("INSERT INTO musician (name) VALUES (%s);", ['Reinhardt'])],
    reverse_sql=[("DELETE FROM musician where name=%s;", ['Reinhardt'])],
)

5、RunPython()

RunSQL() 函数操作的是 SQL 语句,RunPython() 参数则是 Python 函数,可以将我们需要写入的数据都写到函数的步骤里,然后在 RunPython() 中调用

以下是使用示例:

def insert_blog_data(apps, schema_editor):
    Blog = apps.get_model("blog", "Blog")
    db_alias = schema_editor.connection.alias

    Blog.objects.using(db_alias).create(name="name_3", tagline="tagline_3")
    Blog.objects.using(db_alias).create(name="name_4", tagline="tagline_4")



class Migration(migrations.Migration):
    dependencies = [
        ("blog", "0001_initial"),
    ]


    operations = [
        migrations.RunPython(insert_blog_data)
    ]

其中,insert_blog_data 是需要执行的函数,在这个函数里,有两个默认参数,apps 和 schema_editor

apps 可以用来获取我们需要的 model,根据函数 apps.get_model(),

这个函数传入两个参数,一个是 application,我们这里是 blog,

一个 model 的名称,我们这里是 Blog

而 schema_editor 则是可以用于获取数据库的 alias

然后,数据的插入的方式就和普通的 model 的操作方法一致了。

RunPython() 函数和 RunSQL 一样,也可以输入两个参数,第二个参数作用也是用于操作失败的回退操作:

migrations.RunPython(insert_blog_data, reverse_insert)

以上就是介绍 migration 的全部内容了,下一篇笔记将介绍如何在 Django 中使用原生的 SQL 来查询数据。

如果想获取更多后端相关文章,可扫码关注阅读:
image

原文链接:https://www.cnblogs.com/hunterxiong/p/17300914.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Django笔记二十之手动编写migration文件 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • 跟老齐学Python之list和str比较

    跟老齐学Python之list和str比较 在Python中,list和str是两种常见的数据类型。虽然它们都可以存储多个元素,但是它们之间还是有很大区别的。本文将详细讲解list和str的比较,包括定义、操作、转换等方面的内容,并给出两个示例说明。 定义 list和str的定义方式不同。list使用方括号[]来定义,元素之间用逗号隔开。例如: my_lis…

    python 2023年5月13日
    00
  • Python中的集合类型知识讲解

    下面是关于Python中的集合类型的知识讲解,包含两个示例说明。 集合类型的定义 在Python中,集合是一种无序、重复的数据类型,它使用大括号{}或set()函数来定义。下是示例: #大括号定义集合 my_set = {1, 2, 3, 4, 5} # 使用set()函数定义一个集合 my_set set([1, , 3, 4, 5]) 集合类型的特点 集…

    python 2023年5月13日
    00
  • Python实现Telnet自动连接检测密码的示例

    下面是详细的攻略: Python实现Telnet自动连接检测密码的示例 在Python中,我们可以使用telnetlib模块实现Telnet自动连接检测密码的功能。本文将对Python实现Telnet自动连接检测密码的示例进行详细讲解,并提供两个示例说明。 Telnet自动连接检测密码实现过程 在Python中,我们可以使用telnetlib模块实现Teln…

    python 2023年5月14日
    00
  • Python NumPy中的随机数及ufuncs函数使用示例详解

    Python NumPy中的随机数及ufuncs函数使用示例详解 Python NumPy是一种Python开源项目,旨在为Python科学计算提供快速、高效的一个数组库。它包括多维数组对象,以及用于处理这些数组的各种工具。其中之一就是NumPy中的随机数及ufuncs函数。以下是详细讲解: 随机数 生成随机数是一个经常使用的操作,而NumPy中提供了丰富的…

    python 2023年6月3日
    00
  • 怎么把Python添加到环境变量中?

    本文将介绍如何将Python添加到Windows系统的环境变量中。适用的场景包括:安装Python时忘记勾选Add Python.exe to PATH。安装了多个Python版本,需要切换版本时。

    2022年10月30日
    00
  • Python产生Gnuplot绘图数据的方法

    Python可以通过Gnuplot绘图库来进行图形绘制,而Gnuplot本身则可以通过读取格式化的数据文件来生成绘图。因此,我们可以在Python中使用Gnuplot来生成数据文件,进而绘制图形。下面是详细的攻略: 准备工作 要使用Python和Gnuplot进行绘图,需要先安装Gnuplot库和相关的Python库。在Ubuntu Linux系统上,可以使…

    python 2023年6月3日
    00
  • Python内置函数及功能简介汇总

    查看Python内置函数及功能简介汇总可以帮助程序员更深入地了解Python的基础知识,并快速掌握常用的内置函数和方法。以下是具体的攻略: 1. 什么是Python内置函数 Python内置函数是Python解释器预定义的一组函数名称,用于不需要导入模块的情况下使用。这些内置函数有很多种用途,例如字符串、数字、列表等常见数据类型的操作和控制流程的语句等等。 …

    python 2023年5月13日
    00
  • 使用python实现md5加密

    当我们需要进行数据加密时,一种常见的方式是使用哈希算法。其中,MD5算法是一种较为常见且流行的哈希算法,可以使用Python语言轻松实现。以下是使用Python实现MD5加密的完整攻略: 1. 引入hashlib库 Python标准库中提供了hashlib库,它支持多种哈希算法,包括MD5。因此,首先需要引入hashlib库。 import hashlib …

    python 2023年6月2日
    00
合作推广
合作推广
分享本页
返回顶部