查询数据层次图解:若是操做mysql,ORM是在pymysq之上又进行了一层封装前端
#sql中的表 #建立表: CREATE TABLE employee( id INT PRIMARY KEY auto_increment , name VARCHAR (20), gender BIT default 1, birthday DATA , department VARCHAR (20), salary DECIMAL (8,2) unsigned, ); #sql中的表纪录 #添加一条表纪录: INSERT employee (name,gender,birthday,salary,department) VALUES ("alex",1,"1985-12-12",8000,"保洁部"); #查询一条表纪录: SELECT * FROM employee WHERE age=24; #更新一条表纪录: UPDATE employee SET birthday="1989-10-24" WHERE id=1; #删除一条表纪录: DELETE FROM employee WHERE name="alex" #python的类 class Employee(models.Model): id=models.AutoField(primary_key=True) name=models.CharField(max_length=32) gender=models.BooleanField() birthday=models.DateField() department=models.CharField(max_length=32) salary=models.DecimalField(max_digits=8,decimal_places=2) #python的类对象 #添加一条表纪录: emp=Employee(name="alex",gender=True,birthday="1985-12-12",epartment="保洁部") emp.save() #查询一条表纪录: Employee.objects.filter(age=24) #更新一条表纪录: Employee.objects.filter(id=1).update(birthday="1989-10-24") #删除一条表纪录: Employee.objects.filter(name="alex").delete()
一些说明:python
int自增列,必须填入参数 primary_key=True。当model中若是没有自增列,则自动会建立一个列名为id的列。mysql
一个整数类型,范围在 -2147483648 to 2147483647。(通常不用它来存手机号(位数也不够),直接用字符串存,)git
字符类型,必须提供max_length参数, max_length表示字符长度。sql
这里须要知道的是Django中的CharField对应的MySQL数据库中的varchar类型,没有设置对应char类型的字段,可是Django容许咱们自定义新的字段,下面我来自定义对应于数据库的char类型数据库
日期字段,日期格式 YYYY-MM-DD,至关于Python中的datetime.date()实例。django
日期时间字段,格式 YYYY-MM-DD HH:MM[:ss[.uuuuuu]][TZ],至关于Python中的datetime.datetime()实例。后端
用于表示某个字段能够为空。app
若是设置为unique=True 则该字段在此表中必须是惟一的 。
若是db_index=True 则表明着为此字段设置索引。
为该字段设置默认值。
配置auto_now_add=True,建立数据记录的时候会把当前时间添加到数据库。
配置上auto_now=True,每次更新数据记录的时候会更新该字段。
外键类型在ORM中用来表示外键关联关系,通常把ForeignKey字段设置在 '一对多'中'多'的一方。
ForeignKey能够和其余表作关联关系同时也能够和自身作关联关系。
设置要关联的表
设置要关联的表的字段
当删除关联表中的数据时,当前表与其关联的行的行为。
models.CASCADE
删除关联数据,与之关联也删除
是否在数据库中建立外键约束,默认为True。
其他字段参数
models.DO_NOTHING 删除关联数据,引起错误IntegrityError models.PROTECT 删除关联数据,引起错误ProtectedError models.SET_NULL 删除关联数据,与之关联的值设置为null(前提FK字段须要设置为可空) models.SET_DEFAULT 删除关联数据,与之关联的值设置为默认值(前提FK字段须要设置默认值) models.SET 删除关联数据, a. 与之关联的值设置为指定值,设置:models.SET(值) b. 与之关联的值设置为可执行对象的返回值,设置:models.SET(可执行对象)
一对一字段。
一般一对一字段用来扩展已有字段。(通俗的说就是一我的的全部信息不是放在一张表里面的,简单的信息一张表,隐私的信息另外一张表,之间经过一对一外键关联)
设置要关联的表。
设置要关联的字段。
当删除关联表中的数据时,当前表与其关联的行的行为。(参考上面的例子)
用于表示多对多的关联关系。在数据库中经过第三张表来创建关联关系
设置要关联的表
同ForeignKey字段。
同ForeignKey字段。
仅用于多对多自关联时,指定内部是否建立反向操做的字段。默认为True。
举个例子:
class Person(models.Model): name = models.CharField(max_length=16) friends = models.ManyToManyField("self")
此时,person对象就没有person_set属性。
class Person(models.Model): name = models.CharField(max_length=16) friends = models.ManyToManyField("self", symmetrical=False)
此时,person对象如今就可使用person_set属性进行反向查询。
在使用ManyToManyField字段时,Django将自动生成一张表来管理多对多的关联关系。
但咱们也能够手动建立第三张表来管理多对多关系,此时就须要经过through来指定第三张表的表名。
设置关联的字段。
默认建立第三张表时,数据库中表的名称。
class Book(models.Model): title = models.CharField(max_length=32, verbose_name="书名") class Author(models.Model): name = models.CharField(max_length=32, verbose_name="做者姓名") # 本身建立第三张表,分别经过外键关联书和做者 class Author2Book(models.Model): author = models.ForeignKey(to="Author") book = models.ForeignKey(to="Book") class Meta: unique_together = ("author", "book")
class Book(models.Model): title = models.CharField(max_length=32, verbose_name="书名") # 经过ORM自带的ManyToManyField自动建立第三张表 class Author(models.Model): name = models.CharField(max_length=32, verbose_name="做者姓名") books = models.ManyToManyField(to="Book", related_name="authors")
class Book(models.Model): title = models.CharField(max_length=32, verbose_name="书名") # 本身建立第三张表,并经过ManyToManyField指定关联 class Author(models.Model): name = models.CharField(max_length=32, verbose_name="做者姓名") books = models.ManyToManyField(to="Book", through="Author2Book", through_fields=("author", "book")) # through_fields接受一个2元组('field1','field2'): # 其中field1是定义ManyToManyField的模型外键的名(author),field2是关联目标模型(book)的外键名。 class Author2Book(models.Model): author = models.ForeignKey(to="Author") book = models.ForeignKey(to="Book") class Meta: unique_together = ("author", "book")
注意:
当咱们须要在第三张关系表中存储额外的字段时,就要使用第三种方式。
可是当咱们使用第三种方式建立多对多关联关系时,就没法使用set、add、remove、clear方法来管理多对多的关系了,须要经过第三张表的model来管理多对多关系。
ORM对应的类里面包含另外一个Meta类,而Meta类封装了一些数据库的信息。主要字段以下:
ORM在数据库中的表名默认是 app_类名,能够经过db_table能够重写表名。
联合索引。
联合惟一索引。
指定默认按什么字段排序。
只有设置了该属性,咱们查询到的结果才能够被reverse()。
class UserInfo(models.Model): nid = models.AutoField(primary_key=True) username = models.CharField(max_length=32) class Meta: # 数据库中生成的表名称 默认 app名称 + 下划线 + 类名 db_table = "table_name" # 联合索引 index_together = [ ("pub_date", "deadline"), ] # 联合惟一索引 unique_together = (("driver", "restaurant"),) ordering = ('name',) # admin中显示的表名称 verbose_name='哈哈' # verbose_name加s verbose_name_plural=verbose_name
自定义字段在实际项目应用中可能会常常用到,这里须要对他留个印象!
自定义char类型字段:
from django.db import models # Create your models here. #Django中没有对应的char类型字段,可是咱们能够本身建立 class FixCharField(models.Field): ''' 自定义的char类型的字段类 ''' def __init__(self,max_length,*args,**kwargs): self.max_length=max_length super().__init__(max_length=max_length,*args,**kwargs) def db_type(self, connection): ''' 限定生成的数据库表字段类型char,长度为max_length指定的值 :param connection: :return: ''' return 'char(%s)'%self.max_length #应用上面自定义的char类型 class Class(models.Model): id=models.AutoField(primary_key=True) title=models.CharField(max_length=32) class_name=FixCharField(max_length=16) gender_choice=((1,'男'),(2,'女'),(3,'保密')) gender=models.SmallIntegerField(choices=gender_choice,default=3)
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) - 二进制类型
(1)null 若是为True,Django 将用NULL 来在数据库中存储空值。 默认值是 False. (1)blank 若是为True,该字段容许不填。默认为False。 要注意,这与 null 不一样。null纯粹是数据库范畴的,而 blank 是数据验证范畴的。 若是一个字段的blank=True,表单的验证将容许该字段是空值。若是字段的blank=False,该字段就是必填的。 (2)default 字段的默认值。能够是一个值或者可调用对象。若是可调用 ,每有新对象被建立它都会被调用。 (3)primary_key 若是为True,那么这个字段就是模型的主键。若是你没有指定任何一个字段的primary_key=True, Django 就会自动添加一个IntegerField字段作为主键,因此除非你想覆盖默认的主键行为, 不然不必设置任何一个字段的primary_key=True。 (4)unique 若是该值设置为 True, 这个数据字段的值在整张表中必须是惟一的 (5)choices 由二元组组成的一个可迭代对象(例如,列表或元组),用来给字段提供选择项。 若是设置了choices ,默认的表单将是一个选择框而不是标准的文本框,<br>并且这个选择框的选项就是choices 中的选项。
对应关系: 'AutoField': 'integer AUTO_INCREMENT', 'BigAutoField': 'bigint AUTO_INCREMENT', 'BinaryField': 'longblob', 'BooleanField': 'bool', 'CharField': 'varchar(%(max_length)s)', 'CommaSeparatedIntegerField': 'varchar(%(max_length)s)', 'DateField': 'date', 'DateTimeField': 'datetime', 'DecimalField': 'numeric(%(max_digits)s, %(decimal_places)s)', 'DurationField': 'bigint', 'FileField': 'varchar(%(max_length)s)', 'FilePathField': 'varchar(%(max_length)s)', 'FloatField': 'double precision', 'IntegerField': 'integer', 'BigIntegerField': 'bigint', 'IPAddressField': 'char(15)', 'GenericIPAddressField': 'char(39)', 'NullBooleanField': 'bool', 'OneToOneField': 'integer', 'PositiveIntegerField': 'integer UNSIGNED', 'PositiveSmallIntegerField': 'smallint UNSIGNED', 'SlugField': 'varchar(%(max_length)s)', 'SmallIntegerField': 'smallint', 'TextField': 'longtext', 'TimeField': 'time', 'UUIDField': 'char(32)',
choice参数 能够用作 用户性别 用户学历 用户工做状态 客户来源 ... class Userinfo(models.Model): name = models.CharField(max_length=32) password = models.IntegerField(default=123) choice = ( (1,'male'), (2,'female'), (3,'other') ) gender = models.IntegerField(choices=choice) """ 1.若是我存的是上面元组中数字会怎么样 2.若是我存的数字不在元组范围内又会怎样 1.数字没有对应关系 是能够存的 """
前端获取
{{ user_obj.get_gender_display }}
后端获取
from app01 import models ser_obj = models.Userinfo.objects.filter(pk=4).first() print(user_obj.username) print(user_obj.gender) # 针对choices字段 若是你想要获取数字所对应的中文 你不能直接点字段 # 固定句式 数据对象.get_字段名_display() 当没有对应关系的时候 该句式获取到的仍是数字 print(user_obj.get_gender_display())
能够这样使用
record_choices = (('checked', "已签到"), ('vacate', "请假"), ('late', "迟到"), ('noshow', "缺勤"), ('leave_early', "早退"), ) record = models.CharField("上课纪录", choices=record_choices, default="checked", score_choices = ((100, 'A+'), (90, 'A'), (85, 'B+'), (80, 'B'), (70, 'B-'), (60, 'C+'), (50, 'C'), (40, 'C-'), (0, ' D'), (-1, 'N/A'), (-100, 'COPY'), (-1000, 'FAIL'), ) score = models.IntegerField("本节成绩", choices=score_choices, default=-1)
from django.db import models # Create your models here. class Test(models.Model): name = models.CharField(max_length=32, null=True, default=None) age = models.IntegerField(max_length=32, null=True, default=None)
urlpatterns = [ path('admin/', admin.site.urls), path("favicon.ico", RedirectView.as_view(url='static/favicon.ico')), re_path('^index/',views.index), ]
def index(request): product_list_to_insert = list() for i in range(1000): product_list_to_insert.append(models.Test(name='123')) models.Test.objects.bulk_create(product_list_to_insert)
# ############### 添加数据 ############### def index(request): import random product_list_to_insert = list() for x in range(100): #注意下面的添加方式 product_list_to_insert.append(models.Test(name='apollo' + str(x), age=random.randint(18, 89))) models.Test.objects.bulk_create(product_list_to_insert) return render(request, 'index.html')
批量更新数据时,先进行数据过滤,而后再调用update方法进行一次性地更新。下面的语句将生成相似update....frrom....的SQL语句。 # ############### 更新数据 ############### Test.objects.filter(name__contains='apollo1').update(name='Jack')
批量更新数据时,先是进行数据过滤,而后再调用delete方法进行一次性删除。 下面的语句讲生成相似delete from ... where ... 的SQL语句。 # ############### 删除数据 ############### Test.objects.filter(name__contains='jack').delete()
优势:好处在于 django orm会自动帮你建立第三张关系表
缺点:可是它只会帮你建立两个表的关系字段(俩个book_id,author_id) 不会再额外添加字段
虽然方便 可是第三张表的扩展性较差 没法随意的添加额外的字段
class Book(models.Model): ... authors = models.ManyToManyField(to='Author') class Author(models.Models): ...
本身手动添加外键
优势:好处在于第三张表是能够任意的添加额外的字段
缺点:不足之处是在于orm查询的时候,不少方法不支持,查询的时候很是的麻烦
class Book(models.Model): ... class Author(models.Models): ... class Book2Author(models.Model): book_id = models.ForeignKey(to='Book') author_id = models.ForeignKey(to='Author') create_time = models.DateField(auto_now_add=True)
3.半自动(推荐使用******)
手动建表 可是你会告诉orm 第三张表是你本身建的
orm只须要给我提供方便的查询方法
第三种虽然可使用orm查询方法,可是不支持使用下面的方法:
add()
set()
remove()
clear()
class Book(models.Model): ... authors = models.ManyToManyField(to='Author', through='Book2Author', through_fields=('book','author'))#告诉django,是和那个表多对多关联,关联表是什么,关联的字段是什么 class Author(models.Model): ... #books = models.ManyToManyField(to='Book', through='Book2Author', through_fields=('author', 'book')) class Book2Author(models.Model): book = models.ForeignKey(to='Book') author = models.ForeignKey(to='Author') create_time = models.DateField(auto_now_add=True)#创建额外的字段
注意