Python Web编程,Django框架 -- (4) 模型 Model 操作数据库 MySQL

  1. MySQL是Web应用中最常用的数据库, Django模型使用自带的ORM(对象关系映射 Object Relational Mapping)用于实现面向对象编程语言里不同类型系统的数据之间的转换。ORM在业务逻辑层和数据库层之间充当了桥梁的作用。ORM通过使用描述对象和数据库之间的映射的元数据,将程序中的对象自动持久化到数据库中。
image.png
  • 每个模型都是一个Python的类,这些类继承django.db.models.Model

  • 模型类的每个属性都相当于一个数据库的字段

  • 利用这些,Django提供了一个自动生成访问数据库的API

Django项目中 app 文件夹下 models.py 文件中添加如下代码:

from django.db import models
# Create your models here.
class Person(models.Model):
#    id = models.AutoField(primary_key=True)        # 如果没有指定主键,则 Django 会自动执行这么一条语句,自增 主键
    id = models.IntegerField(primary_key=True)
    first_name = models.CharField(max_length=30)
    last_name = models.CharField(max_length=30)

对应的SQL语句为:

CREATE TABLE myapp_persion(
    "id" serial NOT NULL PRIMARY KEY,
    "first_name" varchar(30) NOT NULL,
    "last_name" varchar(30) NOT NULL

);

然后在终端执行如下命令(django项目,manage.py所在的文件夹下执行命令)

$ python manage.py makemigrations

$ python manage.py migrate

ORM 对应关系表

image.png
  1. 连接数据库
  • ORM 无法创建数据库(无法操作数据库级别,只能操作到数据表),所以需要事先创建好数据库。

create database xxx;

  • 修改 settings.py 中的DATABASES配置项。

参考文章: Python Web编程,django框架 -- (2)settings文件详解

  • 重要的一点,这里使用pymysql模块连接mysql数据库,有几个地方需要修改:

  • 在与 settings.py 同级别下的 init.py 中引入模块并进行配置, 因为Django 默认使用的是 MySQLdb,但是 MySQLdb已经不再适用用当前的版本,这里使用 pymysql来替代,需要让Django能够识别 mymysql

(报错信息:django.core.exceptions.ImproperlyConfigured: Error loading MySQLdb module: No module named 'MySQLdb')

import pymysql
pymysql.install_as_MySQLdb()     # 加载这条语句后, 程序如果再使用 import MySQLdb, 实际会使用pymysql。  sys.modules["MySQLdb"] = sys.modules["_mysql"] = sys.modules["pymysql"]
  • 由于导入的pymysql与原有django需要使用的MySQLdb版本不匹配,所以会有版本信息报错,解决方法是注释掉版本报错信息。

(报错信息:mysqlclient 1.3.13 or newer is required: ....)

可以通过点击报错信息中指向的链接打开对应的 python 文件(路径为 ..\Anaconda3\envs\learn\Lib\site-packages\django\db\backends\mysql\base.py,红色部分 要根据实际配置的文件路径查找)

查找到如下的代码,并注释掉:

# if version < (1, 3, 13):
#     raise ImproperlyConfigured('mysqlclient 1.3.13 or newer is required; you have %s.' % Database.__version__)
  • 字符串报错,这是由于早期的 Python 版本需要对字符串进行 encode 及 decode 转换,现在已经不需要了。根据报错信息找到对应的文件,将对应的代码注释掉。

(报错信息:AttributeError: 'str' object has no attribute 'decode')

文件路径:...\Anaconda3\envs\learn\Lib\site-packages\django\db\backends\mysql\operations.py

def last_executed_query(self, cursor, sql, params):
    query = getattr(cursor, '_executed', None)
#     if query is not None:
#         query = query.decode(errors='replace')
    return query
  1. 数据库的增删改查

Django 提供了 shell 功能,可以单独操作数据库。

打开命令行,找到 manage.py 所在的文件夹:

$ python manage.py shell

这样就进入了shell功能,就是进入了python应用程序,同时包含了 django 的一些设置。

  • 创建数据库:

在有了继承自 models.Model 的类(相当于创建了数据库的表格)之后,实例化这个类的操作就相当于添加数据库的记录。

首先在 models.py 中添加如下代码,创建数据库的表格,并执行 makemigrations, migrate 命令 完成数据库表格的创建:

from django.db import models
# Create your models here.
class Type(models.Model):
    typename = models.CharField(max_length=20)
class Name(models.Model):

    # id 自动创建

    name = models.CharField(max_length=50)

    author = models.CharField(max_length=50)

    stars = models.CharField(max_length=5)
  • 然后在 shell 环境中 导入上述类,并执行 记录添加操作, 最后通过 save() 功能写入 数据库。

如下:


image.png

到数据库中查询是否已经添加成功:

image.png
  • 使用Django自带 ORM 框api实现:
image.png

数据库查询结果:

image.png
  • 除了在 shell 中实时操作数据库,也可以在 django项目中操作数据库,首先需要完成 django项目的设置,url 的建立,然后在views中创建视图函数来操作数据库,建立 url 到这个视图函数的关系,这样,每一次打开网页(即执行操作数据库的视图函数)就完成一次实际的数据库操作(创建或查询数据)

在 views 文件中添加如下代码:

from django.shortcuts import render
from django.http.response import HttpResponse
from .models import Name

# Create your views here.
def testdb(request):
    n = Name()
    n.name = '红楼梦'
    n.author = '曹雪芹'
    n.stars = 9.6
    n.save()
    return HttpResponse('<p>数据添加成功!</p>')

执行 python manage.py runserver,运行 django项目,然后打开网页,打开对应的url,每一次打开网页就将创建一条记录。

  • 查询数据库

以下为常见的查询语句,其中除了 get 是直接返回对象,其他的都是返回 QuerySet对象,可以继续增加条件查询。

from django.shortcuts import render
from django.http.response import HttpResponse
from .models import Name

# Create your views here.
def testdb(request):
    n = Name()
    n.name = '红楼梦'
    n.author = '曹雪芹'
    n.stars = 9.6
    n.save()
    return HttpResponse('<p>数据添加成功!</p>')
def querydb(request):

    # objects 模型管理器是操作数据库的入口,其 all() 方法可以获得所有的数据行,相当于 SQL 的 SELECT * FROM ...
    list = Name.objects.all()

    # filter 相当于 SQL 的where,设置过滤条件
    result1 = Name.objects.filter(id=2)
    # --> <QuerySet [<Name: Name object (2)>]>

    # get 获取单个对象
    result2 = Name.objects.get(id=6)
    # --> Name object (6)

    # 数据排序 order_by
    result3 = Name.objects.order_by('name')
    # --> <QuerySet [<Name: Name object (6)>, <Name: Name object (2)>, <Name: Name object (8)>, <Name: Name object (9)>, <Name: Name object (10)>, <Name: Name object (7)>]>

    # 限制返回数据,相当于 limit
    result4 = Name.objects.order_by('name')[0:2]

    # 组合使用
    # gt, gte, lt, lte 代表 大于,大于等于,小于,小于等于
    result5 = Name.objects.filter(id__lte=6).order_by('id')

    # Values_list  获取数据:
    Name.objects.values_list('name')

    # -- > <QuerySet [('红楼梦',), ('活着',), ('西游记',), ('红楼梦',), ('红楼梦',), ('红楼梦',)]>
    Name.objects.values_list()

    # -- >  <QuerySet [(2, '红楼梦', '曹雪芹', '9.6'), (6, '活着', '余华', '9.3'), (7, '西游记', '吴承恩', '9.2'), (8, '红楼梦', '曹雪芹', '9.6'), (9, '红楼梦', '曹雪芹', '9.6'), (10, '红楼梦', '曹雪芹', '9.6')]>
    response_all = [var.name for var in list]
    for i in range(1, 6):
        print(eval('result' + str(i)))
    response = ','.join(response_all)
    return HttpResponse('<p>' + response + '</p>')

*   更新数据库, 可以用save() 或 update()
def updatedb(request):
    # 先获取实例对象,修改后再保存
    n = Name.objects.get(id=2)
    n.name = '石头记'
    n.save()

    # 另一种方式修改数据
    Name.objects.filter(id=7).update(name='测试')

    # 修改全部数据
    Name.objects.all().update(name='测试')
    return HttpResponse('<p>修改成功</p>')

*   删除数据库
def deletedb(request):
    obj = Name.objects.get(id=8)
    obj.delete()
    Name.objects.filter(id=8).delete()
    Name.objects.all().delete()
    return HttpResponse('<p>删除成功</p>')

参考资料:

  1. Django模型:https://www.runoob.com/django/django-model.html

  2. django数据库过滤的总结:https://www.cnblogs.com/pythonyeyu/p/10220093.html

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容