欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

django中的ORM

程序员文章站 2024-02-10 23:32:28
...

首先要在settings.py文件中配置数据库信息

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.mysql',  # 数据库类型
        'NAME': 'day63',    # 数据库的名字
        'HOST': '127.0.0.1',  # ip地址
        'PORT': 3306,  # 端口,默认是3306
        'USER': 'root',  # 用户类型
        'PASSWORD': '123456', 
    }
}
# 注意 Django中的ORM只能操作数据表 ,不能操作数据库,所以数据库还是要自己建的

然后 在项目的__init__.py文件中写如下代码

import pymysql
# 告诉Django我用的是MySQL数据库
pymysql.install_as_MySQLdb()

在Django中有如下关系:
Python类 ----> 数据表
对象 ----> 数据行 (一个表中有很多条数据,每一条数据 就是一个对象)
属性 ----> 字段 (比如一个对象 有id值 有name)
下面是一个例子:

class Author(models.Model):  # 新建一张作者表
    id = models.AutoField(primary_key=True)  # 设置id列为自增列 且为主键
    name = models.CharField(max_length=32) # CharField 是 varchar 类型
    books = models.ManyToManyField(to="Book")  #  多对多关系 (一个作者可以有多本书,一本书可以有多个作者)

这样 就定义好了一张表,接下去我们要拿着这个类去数据库中创建这张表:
首先要打开终端 (必须在项目的跟目录下),执行以下语句:
1. python manage.py makemigrations – > 把models.py里面的更改记录到小本本上
2. python manage.py migrate – > 把更改翻译成SQL语句执行
这里要注意的是,表名并不是author,而是app_author,如果要自定表名,请看下面的例子:

    class Author(models.Model):  # 新建一张作者表
        id = models.AutoField(primary_key=True)  # 设置id列为自增列 且为主键
        name = models.CharField(max_length=32) # CharField 是 varchar 类型
        books = models.ManyToManyField(to="Book")  #  多对多关系 (一个作者可以有多本书,一本书可以有多个作者)
        
	    class Meta:
	        db_table = "author"  # 自定义表名为author 

修改了models里的类之后 要重新执行上面两条语句
1. python manage.py makemigrations – > 把models.py里面的更改记录到小本本上
2. python manage.py migrate – > 把更改翻译成SQL语句执行
这样我们就成功修改了表名
其他说明 :
id字段是自动添加的(即便你不写id字段,Django会自动给你加上),如果你想要指定自定义主键,只需在其中一个字段中指定 primary_key=True 即可。如果Django发现 你已经明确地设置了Field.primary_key,它将不会添加自动ID列。

ORM中的常用字段

1.AutoField

int类型的自增列,必须填入参数 primary_key=True。当model中如果没有自增列,则自动会创建一个列名为id的列

2.IntegerField

一个整数类型,范围在 -2147483648 to 2147483647。

3.CharField

字符类型,必须提供max_length参数, max_length表示字符长度。

4.DateField

日期字段,日期格式 YYYY-MM-DD,相当于Python中的datetime.date()实例。

5.DateTimeField

日期时间字段,格式 YYYY-MM-DD HH:MM[:ss[.uuuuuu]][TZ],相当于Python中的datetime.datetime()实例。

字符合集(好吧,我承认我记不住)

AutoField(Field)
- int自增列,必须填入参数 primary_key=True

BigAutoField(AutoField)
- bigint自增列,必须填入参数 primary_key=True

注:当model中如果没有自增列,则自动会创建一个列名为id的列
from django.db import models

class UserInfo(models.Model):
# 自动创建一个列名为id的且为自增的整数列
username = models.CharField(max_length=32)

class Group(models.Model):
# 自定义自增列
nid = models.AutoField(primary_key=True)
name = models.CharField(max_length=32)

SmallIntegerField(IntegerField):
- 小整数 -32768 ~ 32767

PositiveSmallIntegerField(PositiveIntegerRelDbTypeMixin, IntegerField)
- 正小整数 0 ~ 32767
IntegerField(Field)
- 整数列(有符号的) -2147483648 ~ 2147483647

PositiveIntegerField(PositiveIntegerRelDbTypeMixin, IntegerField)
- 正整数 0 ~ 2147483647

BigIntegerField(IntegerField):
- 长整型(有符号的) -9223372036854775808 ~ 9223372036854775807

BooleanField(Field)
- 布尔值类型

NullBooleanField(Field):
- 可以为空的布尔值

CharField(Field)
- 字符类型
- 必须提供max_length参数, max_length表示字符长度

TextField(Field)
- 文本类型

EmailField(CharField):
- 字符串类型,Django Admin以及ModelForm中提供验证机制

IPAddressField(Field)
- 字符串类型,Django Admin以及ModelForm中提供验证 IPV4 机制

GenericIPAddressField(Field)
- 字符串类型,Django Admin以及ModelForm中提供验证 Ipv4和Ipv6
- 参数:
protocol,用于指定Ipv4或Ipv6, ‘both’,“ipv4”,“ipv6”
unpack_ipv4, 如果指定为True,则输入::ffff:192.0.2.1时候,可解析为192.0.2.1,开启此功能,需要protocol=“both”

URLField(CharField)
- 字符串类型,Django Admin以及ModelForm中提供验证 URL

SlugField(CharField)
- 字符串类型,Django Admin以及ModelForm中提供验证支持 字母、数字、下划线、连接符(减号)

CommaSeparatedIntegerField(CharField)
- 字符串类型,格式必须为逗号分割的数字

UUIDField(Field)
- 字符串类型,Django Admin以及ModelForm中提供对UUID格式的验证

FilePathField(Field)
- 字符串,Django Admin以及ModelForm中提供读取文件夹下文件的功能
- 参数:
path, 文件夹路径
match=None, 正则匹配
recursive=False, 递归下面的文件夹
allow_files=True, 允许文件
allow_folders=False, 允许文件夹

FileField(Field)
- 字符串,路径保存在数据库,文件上传到指定目录
- 参数:
upload_to = “” 上传文件的保存路径
storage = None 存储组件,默认django.core.files.storage.FileSystemStorage

ImageField(FileField)
- 字符串,路径保存在数据库,文件上传到指定目录
- 参数:
upload_to = “” 上传文件的保存路径
storage = None 存储组件,默认django.core.files.storage.FileSystemStorage
width_field=None, 上传图片的高度保存的数据库字段名(字符串)
height_field=None 上传图片的宽度保存的数据库字段名(字符串)

DateTimeField(DateField)
- 日期+时间格式 YYYY-MM-DD HH:MM[:ss[.uuuuuu]][TZ]

DateField(DateTimeCheckMixin, Field)
- 日期格式 YYYY-MM-DD

TimeField(DateTimeCheckMixin, Field)
- 时间格式 HH:MM[:ss[.uuuuuu]]

DurationField(Field)
- 长整数,时间间隔,数据库中按照bigint存储,ORM中获取的值为datetime.timedelta类型

FloatField(Field)
- 浮点型

DecimalField(Field)
- 10进制小数
- 参数:
max_digits,小数总长度
decimal_places,小数位长度

BinaryField(Field)
- 二进制类型

自定义字段

除了上面的一坨,还有自定义字段类型,这个就比较重要了,可能会面试到哦

class FixedCharField(models.Field):
    """
    自定义的char类型的字段类
    """
    def __init__(self, max_length, *args, **kwargs):
        super().__init__(max_length=max_length, *args, **kwargs)
        self.length = max_length

    def db_type(self, connection):
        """
        限定生成数据库表的字段类型为char,长度为length指定的值
        """
        return 'char(%s)' % self.length

定义好了以后,可以用在其他表中

class Person(models.Model):
    name = models.CharField(max_length=32)
    new_name = FixedCharField(max_length=64, default="佩奇")   # 看我看我!!
    age = models.IntegerField(default=18)
    birthday = models.DateField(auto_now_add=True)
    class Meta:
    	db_table = "person"

然后神奇的事发生了
django中的ORM

字段参数 (这个好理解,简单略过。。)

  1. null 用于表示某个字段可以为空。
  2. unique 如果设置为unique=True 则该字段在此表中必须是唯一的 。
  3. db_index 如果db_index=True 则代表着为此字段设置数据库索引
  4. default 为该字段设置默认值。

时间字段独有

  1. auto_now_add
    配置auto_now_add=True,创建数据记录的时候会把当前时间添加到数据库
  2. auto_now
    配置上auto_now=True,每次更新数据记录的时候会更新该字段。

关系字段 ·ForeignKey

字段参数

to : 设置要关联的表
to_field :设置要关联的表的字段
related_name:反向操作时,使用的字段名,用于代替原反向查询时的’表名_set’。 – >这个我也不会。。。
但是 但是 但是!!! 有例子。。

 class Classes(models.Model):
    name = models.CharField(max_length=32)

class Student(models.Model):
    name = models.CharField(max_length=32)
    theclass = models.ForeignKey(to="Classes")

当我们要查询某个班级关联的所有学生(反向查询)时,我们会这么写:

models.Classes.objects.first().student_set.all()

当我们在ForeignKey字段中添加了参数 related_name 后,

class Student(models.Model):
    name = models.CharField(max_length=32)
    theclass = models.ForeignKey(to="Classes", related_name="students")

当我们要查询某个班级关联的所有学生(反向查询)时,我们会这么写:

models.Classes.objects.first().students.all()

嗯,今天就更那么多,其他的我不会了。。。