6 单表操作
一、操作单表之前的简单配置
1 创建表
在 Django 项目下对应的 app 目录中的 models.py 文件下创建模型
(1)创建模型
from django.db import models class Book(models.Model): id=models.AutoField(primary_key=True) name=models.CharField(max_length=32) price=models.DecimalField(max_digits=5,decimal_places=2) publish_date=models.DateField(default='1999-11-27') publish=models.ForeignKey(to=Publish,to_field='id')
2 在 settings 中选择并设置需要的数据库
DATABASES = { 'default': { 'ENGINE': 'django.db.backends.mysql', 'NAME': '连接的数据库名称', 'USER': '账户名称', 'PASSWORD': '账户密码', 'HOST': '127.0.0.1', 'PORT': 3306, 'ATOMIC_REQUEST': True, 'OPTIONS': { "init_command": "SET storage_engine=MyISAM", } } } ''' 'NAME':要连接的数据库,连接前需要创建好 'USER':连接数据库的用户名 'PASSWORD':连接数据库的密码 'HOST':连接主机,默认本机 'PORT':端口 默认3306 'ATOMIC_REQUEST': True, 设置为True统一个http请求对应的所有sql都放在一个事务中执行(要么所有都成功,要么所有都失败)。 是全局性的配置, 如果要对某个http请求放水(然后自定义事务),可以用non_atomic_requests修饰器 'OPTIONS': { "init_command": "SET storage_engine=MyISAM", } 设置创建表的存储引擎为MyISAM,INNODB '''
1 在 mysql 连接前该数据库必须已经创建,而上面的 sqlite 数据库下的 db.sqlite3 则是项目自动创建 USER 和 PASSWORD 分别是数据库的用户名和密码。设置完后,再启动我们的Django 项目前,我们需要激活我们的 mysql 。然后,启动项目,会报错:no module named MySQLdb 。这是因为 django 默认你导入的驱动是 MySQLdb,可是MySQLdb 对于py3有很大问题,所以我们需要的驱动是PyMySQL 所以,我们只需要找到项目名文件下的__init__,在里面写入: 2 3 import pymysql 4 pymysql.install_as_MySQLdb()
3 执行数据库迁移命令即可创建出对应的表
(1)方式一:
1 初始化数据,此时还未开始创建或者更新数据 python manage.py makemigrations 2 开始创建 python manage.py migrate
(2)方式二:
进入上述图片的操作之后,会进入一个类似于命令行的窗口,直接执行命令 (1)makemigrations 和 (2)migrate 两条命令后即可创建表格
4 INSTALLED_APPS 确保配置文件中写入创建的 app 名称
INSTALLED_APPS = [ 'django.contrib.admin', 'django.contrib.auth', 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.messages', 'django.contrib.staticfiles', 'app01.apps.App01Config', # app01,这个不是建议写法 ]
注意:
MySQLclient目前只支持到python3.4,因此如果使用的更高版本的python,需要修改如下:
通过查找路径C:\Programs\Python\Python36-32\Lib\site-packages\Django-2.0-py3.6.egg\django\db\backends\mysql 这个路径里的文件把 if version < (1, 3, 3): raise ImproperlyConfigured("mysqlclient 1.3.3 or newer is required; you have %s" % Database.__version__)
二、单表操作
1 字段的增加、删除
(1)删除:直接注释掉字段,执行数据库迁移命令即可
(2)新增字段:在 models.py 类里直接新增字段,直接执行数据库迁移命令会提示输入默认值,此时需要设置
publish = models.CharField(max_length=32,default='默认值',null=True)
注意:
1 数据库迁移记录都在 app01下的migrations里 2 使用showmigrations命令可以查看没有执行migrate的文件 3 makemigrations是生成一个文件,migrate是将更改提交到数据量
2 添加表记录
(1)方式一:
# create 方法的返回值 book_obj 就是插入 book 表中的 Django 这本书籍纪录对象 book_obj=Book.objects.create(name="Django",state=True,price=998,publish="未知出版社",pub_date="9999-99-99")
(2)方式二:
book_obj=Book(name="Django",state=True,price=998,publish="未知出版社",pub_date="9999-99-99") book_obj.save()
3 查询表记录的常用 API
(1) all(): 查询所有结果 (2) filter(**kwargs): 它包含了与所给筛选条件相匹配的对象 (3) get(**kwargs): 返回与所给筛选条件相匹配的对象,返回结果有且只有一个,如果符合筛选条件的对象超过一个或者没有都会抛出错误。 (4) exclude(**kwargs): 它包含了与所给筛选条件不匹配的对象 (5) order_by(*field): 对查询结果排序,默认是从大到小排序;字符串前加 - 号('-id'),表示从小到大排序 (6) reverse(): 对查询结果反向排序 (7) count(): 返回数据库中匹配查询(QuerySet)的对象数量。 (8) first(): 返回第一条记录 (9) last(): 返回最后一条记录 (10) exists(): 如果QuerySet包含数据,就返回True,否则返回False (11) values(*field): 返回一个ValueQuerySet——一个特殊的QuerySet,运行后得到的并不是一系列 model 的实例化对象,而是一个可迭代的字典序列 (12) values_list(*field): 它与values()非常相似,它返回的是一个元组序列,values返回的是一个字典序列 (13) distinct(): 从返回结果中剔除重复纪录
1 book = Book.objects.all().gei(id="2") # get 只能用来查询返回一条数据的情况,记录过多或者过少都会产生报错 2 book_dic = Book.objects.all().values('name','price') # 返回一个类似字典的 QuerySet 对象,经过 first() 处理之后就不是对象了,可以通过循环来取值 3 book_tuple = Book.objects.all().values_list() # 返回一个类似元组的 QruerySet 对象,可循环取值 4 reverse() # 将查询结果反向排序,但是必须在 order_by() 之后使用才可以生效 5 1️⃣ Book.objects.all().order_by('price','publish_date').reverse() 6 2️⃣ Book.objects.all().order_by('-price','publish_date') 7 1️⃣ 和 2️⃣ 结果是相同的
4 双下划线的模糊查询
1 Book.objects.filter(price__in=[100,200,300]) 2 Book.objects.filter(price__gt=100) # 大于100 3 Book.objects.filter(price__lt=100) 4 Book.objects.filter(price__gte=100) # 大于等于100 5 Book.objects.filter(price__lte=100) 6 Book.objects.filter(price__range=[100,200]) # 在 100 至 200 之间,包括首尾 7 Book.objects.filter(name__contains="Django") 8 Book.objects.filter(name__icontains="Django") # 忽略大小写,由于版本原因,结果可能不一样 9 Book.objects.filter(name__startswith="D") 10 Book.objects.filter(publish_date__year=9999) # 出版日期的年份,对应的还有月份日期等
5 删除记录
1️⃣ Book.objects.filter(name="Django").first().delete() # 注意是没有返回值的 2️⃣ QuerySet.delete() 1️⃣ 和 2️⃣ 的结果是一样的 (1,{'app01.Book':1}) # 各个参数的说明 第一个 1 是:总体影响的行数 app01.Book:哪个表的记录 第二个 1 是:影响表的记录
delete() 方法是 QuerySet 上的方法,但并不适用于 Manager 本身。这是一种保护机制,是为了避免意外地调用
Book.objects.delete() 方法导致 所有的 记录被误删除。如果你确认要删除所有的对象,那么你必须显式地调用:
Book.objects.all().delete()
若不想级联删除,可以设置为: pubHouse = models.ForeignKey(to='Publisher', on_delete=models.SET_NULL, blank=True, null=True)
在 Django 删除对象时,会模仿 SQL 约束 ON DELETE CASCADE 的行为,换句话说,删除一个对象时也会删除与它相关联的外键对象。例如: b = Blog.objects.get(pk=1) # This will delete the Blog and all of its Entry objects. b.delete()
6 更新表记录
update() 方法对于任何结果集(QuerySet)均有效,Manage 对象不可调用,意味着你可以同时更新多条记录update()方法会返回一个整型数值,表示受影响的记录条数。
Book.objects.filter(name__startswith="D").update(price=999)
三、在 Python 脚本中调用 Django 环境(了解)
import os if __name__ == '__main__': os.environ.setdefault("DJANGO_SETTINGS_MODULE", "Django名称.settings") import django django.setup() from app01 import models books = models.Book.objects.all() print(books)
注意:
Django 名称可以在 manage.py 中查看或者核对
四、Django 中打印 SQL 语句
方式一: LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'handlers': { 'console':{ 'level':'DEBUG', 'class':'logging.StreamHandler', }, }, 'loggers': { 'django.db.backends': { 'handlers': ['console'], 'propagate': True, 'level':'DEBUG', }, } } 方式二: obj = Book.objects.all().filter(pk=1) print(obj.query)