Django之数据库对象关系映射
Django ORM基本配置
到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞:
- 创建数据库,设计表结构和字段
- 使用 MySQLdb 来连接数据库,并编写数据访问层代码
- 业务逻辑层去调用数据访问层执行数据库操作
django为使用一种新的方式,即:关系对象映射(Object Relational Mapping,简称ORM),django中遵循 Code Frist 的原则,即:根据代码中定义的类来自动生成数据库表;
1、修改project数据库配置(程序主目录下的settings.py文件)
默认连接数据库为本地文件sqlite3:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.sqlite3',
'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
}
}
更换为指定的mysql数据库:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.mysql',
'NAME': 'mysite', #一定要存在的数据库名
'USER': 'xxxx',
'PASSWORD': 'xxx',
'HOST': '192.168.xx.xx',
'PORT': ''
}
}
2、创建定义数据库表结构文件(对应app目录下的models.py文件)
生成一个简单的数据库表:
from django.db import models # Create your models here. class UserInfo(models.Model): username = models.CharField(max_length=32)
passwd = models.CharField(max_length=64)
把对应的app名称加入到settings.py文件配置里:
INSTALLED_APPS = [
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'cmdb02',
]
3、生成数据库表
执行下面命令:
python manage.py makemigrations
python manage.py migrate # 生成数据表
注意:Django默认用的MysqlDB模块连接数据库,但在python3.x里面还没有这个模块,所有需要把连接数据库的模块改成pymsyql,修改project目录下的init.py文件
import pymysql
pymysql.install_as_MySQLdb()
执行生成数据库表命令,遇到的问题:
a. django.core.exceptions.ImproperlyConfigured: mysqlclient 1.3.13 or newer is required; you have 0.9.3.
解决: 找到 python_path\Lib\site-packages\django\db\backends\mysql路径下的base.py, 注释以下代码:
b. AttributeError: 'str' object has no attribute 'decode'
解决: 找到 python_path\Lib\site-packages\django\db\backends\mysql路径下的operations.py, 修改以下代码:
query = query.encode(errors='replace') 将decode修改为encode
c. pymysql.err.InternalError: (1049, "Unknown database 'mysite'")
解决: 数据库创建数据库 mysite
django数据库增删改查,可以直接运行python文件,具体配置如下:
新建python文件,设置django的配置:
import django, os
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'dj_test.settings') #设置django的配置文件
django.setup() from user import models
from django.db.models import Q #新增
# models.Nav.objects.create(name='哈哈哈') # 对象实例化
# nav = models.Nav(name='我的心情')
# nav.save()
4. Django数据库单表操作
1. 增加:
第一种写法:
def ormadd(request):
UserInfo.objects.create(username='root',passwd='')
return HttpResponse('orm add')
第二种写法:
def ormadd(request):
dicts = {'username': "xiaoxiao", 'passwd': ''}
UserInfo.objects.create(**dicts)
return HttpResponse('orm add')
第三种写法:
def ormadd(request):
userinfo = UserInfo(username='sb2',passwd='')
userinfo.save()
return HttpResponse('orm add')
2.删除数据
def ormdel(request):
UserInfo.objects.filter(id=19).delete()
return HttpResponse('orm dele')
3.更新数据
第一种写法:
def ormadd(request):
UserInfo.objects.filter(id__gt=10).update(username='white')
#id大于10的数据,更新name为101
return HttpResponse('orm update')
第二种写法:
def ormadd(request):
dicts ={'username': 'black'}
UserInfo.objects.filter(id__gt=10).update(**dicts)
return HttpResponse('orm update')
4.查询数据
- 查询所有的数据
def ormadd(request):
res = UserInfo.objects.all() #QuerySet类型,列表里每个元素都是obj对象
print(res) # <QuerySet [<UserInfo: UserInfo object>, <UserInfo: UserInfo object>, <UserInfo: UserInfo object>]>
for row in res:
# 1 root 123456
# 2 admin 123123
print(row.id, row.username, row.passwd)
return HttpResponse('orm select')
2.查询指定字段
def ormadd(request):
res = UserInfo.objects.filter(username='root') #过滤后,结果为list
#res = UserInfo.objects.filter(id=3) #根据id查询
new_res = {}
if res:
for row in res:
new_res['id'] = row.id
new_res['username'] = row.username
new_res['passwd'] = row.passwd
return render(request, 'login.html', {'new_res':new_res})tmplate页面数据显示:
3. 获取查询第一条数据 和 统计匹配个数
def ormadd(request):
#获取匹配的第一条数据
obj = UserInfo.objects.filter(username='root').first()
#获取匹配的字段个数
c = UserInfo.objects.filter(username='root').count()
return render(request, 'login.html', {'obj': obj, 'c': c})template页面数据显示:
4. 比较值查询及多条件查询
def ormadd(request):
UserInfo.objects.filter(id=3, username='root') #id=1 且 name=root
UserInfo.objects.filter(id__gt=1, username='root') #id>1 且 name=root
UserInfo.objects.filter(id__lt=1) #id<1
UserInfo.objects.filter(id__gte=1) #id>=1
UserInfo.objects.filter(id__lte=1) #id<=1
UserInfo.objects.filter(username__contains='root') #模糊查询
UserInfo.objects.filter(id__range=(1,3)) #在什么范围
UserInfo.objects.filter(id__in=[1,2,3,4,5,6]) #在什么范围
UserInfo.objects.exclude(id=1) #排除id=1的数据
from django.db.models import Q
UserInfo.objects.filter(Q(username__contains='root')|Q(id__gte=1)) #username包含root 或者 id>=1的数据 或者关系
5.外键反向查询
#导航表结构
class Nav(models.Model):
name = models.CharField(max_length=64, unique=True, verbose_name='导航名称')
is_delete = models.SmallIntegerField(default=1,verbose_name='是否被删除') #0已删
create_time = models.DateTimeField(verbose_name='创建时间',auto_now_add=True) #插入数据自动转换为当前时间
update_time = models.DateTimeField(verbose_name='更新时间', auto_now=True) #修改时间自动转换为当前时间 def __str__(self):
return self.name class Meta:
verbose_name = '导航表'
verbose_name_plural = verbose_name
db_table = 'nav' #指定表名
# ordering=['create_time'] #查询数据时,默认按照 某个字段排序 #文章表结构
class Article(models.Model):
title = models.CharField(max_length=20, verbose_name='文章名称')
content = models.TextField(verbose_name='文章内容',null=True)
img = models.ImageField(upload_to='article_img',verbose_name='文章图片',null=True) #指定上传到哪个目录下
nav = models.ForeignKey(Nav,verbose_name='导航表',on_delete=models.DO_NOTHING,db_constraint=False) #外键,对应导航表的数据删除后,该表不需要删除; db_contraint不建立真正的外键关系
is_delete = models.SmallIntegerField(default=1, verbose_name='是否被删除')
create_time = models.DateTimeField(verbose_name='创建时间', auto_now_add=True) # 插入数据自动转换为当前时间
update_time = models.DateTimeField(verbose_name='更新时间', auto_now=True) # 修改时间自动转换为当前时间 def __str__(self):
return self.title class Meta:
db_table='article' #外键反向查询
nav = models.Nav.objects.get(name='python')
res_a = nav.article_set.all() #查导航下所有的文章
print(res_a)
6. 多对多关联 表结构
表设计:
from django.db import models
from utils import tools
from earth import settings class BaseModel(models.Model):
'''公共字段'''
is_delete_choice = (
(0, '删除'),
(1, '正常')
)
is_delete = models.SmallIntegerField(choices=is_delete_choice, default=1, verbose_name='是否被删除')
create_time = models.DateTimeField(verbose_name='创建时间', auto_now_add=True) # auto_now_add的意思,插入数据的时候,自动取当前时间
update_time = models.DateTimeField(verbose_name='修改时间', auto_now=True) # 修改数据的时候,时间会自动变 class Meta:
abstract = True # 只是用来继承的,不会创建这个表 class Author(BaseModel):
name = models.CharField(verbose_name='名称', max_length=20) def __str__(self):
return self.name class Meta:
verbose_name = '作家'
verbose_name_plural = verbose_name
ordering = ['id']
db_table = 'eg_author' class Book(BaseModel):
name = models.CharField(verbose_name='书名', max_length=20)
price = models.FloatField(verbose_name='价格')
count = models.IntegerField(verbose_name='数量')
# author = models.ForeignKey(Author, on_delete=models.DO_NOTHING, db_constraint=False, verbose_name='作者')
author = models.ManyToManyField(Author, verbose_name='作者') #多对多关联, 1个作者可以有多本书; 1本书可以有多个作者翻译 def __str__(self):
return self.name class Meta:
verbose_name = '书籍'
verbose_name_plural = verbose_name
ordering = ['id']
db_table = 'eg_book'
model_test.py, 多对多关联查询(正向、反向)
import django,os
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'earth.settings') #设置django的配置文件
django.setup() from example import models #新增作者
models.Author.objects.create(name='dsx')
models.Author.objects.create(name='niuniu') #新增书
models.Book.objects.create(name='自动化1',price=18,count=2)
models.Book.objects.create(name='自动化2',price=18,count=3)
models.Book.objects.create(name='自动化3',price=18,count=4) #获取2张表的数据
a1 = models.Author.objects.filter(id=4).first()
a2 = models.Author.objects.filter(id=5).first()
b1 = models.Book.objects.filter(id=9).first()
b2 = models.Book.objects.filter(id=10).first() #创建多对多关系,以Book表为基础
b1.author.add(a1)
b1.author.add(a2)
b2.author.add(a1)
b2.author.add(a2) #多对多关系查询,正向查询
a = b1.author.all() #b1这本书对应的作者,正向查询
print(a) #反向查询
b = a1.book_set.all() # 根据a1作者查询有几本书,反向查询
print(b) #解除多对多关系
b1.author.remove(a1) #解除书1与作者1的关系 #清除关系
b2.author.clear() #清除b2这本书对应的所有作者
django 分页:
from django.core.paginator import Paginator
#整体分页功能
m = list(range(100))
page_obj = Paginator(m,20)
print(page_obj.count) #总共多少数据
print(list(page_obj.get_page(1))) #获取第几页数据
print(page_obj.num_pages) #总共分几页 ,100/20 = 5页
print(page_obj.page_range) #分页范围,默认分页范围 #某页的功能
page1 = page_obj.get_page(1)
page1.has_next() #判断是否有下一页
page1.has_other_pages() #是否有其他页
page1.has_previous() #是否有上一页
page1.next_page_number() #下一页的页数
page1.previous_page_number() #上一页页码
page1.end_index() #末尾页
page1.start_index() #首页
page1.number #当前页码
page1.paginator #获取分页对象
django查询count、group_by功能
SQL:select department,count(*) from EmployeeInfo group by department; from django.db.models import Count result = EmployeeInfo.objects.values('department').annotate(Count=Count('department')).order_by()
#例子:
html:
{% block pagination %}
{# 判断是否有其他分页#}
{% if artciles.has_other_pages %}
<div>
<ul class="pagination">
{# 判断是否有上一页#}
{% if artciles.has_previous %}
<li><a href="/index/?limit={{ page_limit }}&page={{ artciles.previous_page_number }}">«</a></li>
{% endif %} {# 分页范围#}
{% for num in artciles.paginator.page_range %}
{% if num == artciles.number %}
<li><a class="active" href="/index/?limit={{ page_limit }}&page={{ num }}">{{ num }}</a></li>
{% else %}
<li><a href="/index/?limit={{ page_limit }}&page={{ num }}">{{ num }}</a></li>
{% endif %}
{% endfor %} {# 判断是否有下一页#}
{% if artciles.has_next %}
<li><a href="/index/?limit={{ page_limit }}&page={{ artciles.next_page_number }}">»</a></li>
{% endif %}
</ul>
</div>
{% endif %}
{% endblock %}
view 代码:
def index(request):
limit = request.GET.get('limit', page_limit)
page = request.GET.get('page', 1) artciles = models.Article.objects.all().order_by('id') #查mysql
page_obj = Paginator(artciles, limit) #对文章进行分页, (7,2)
page_data = page_obj.get_page(page) #获取某页数据
return render(request, 'index.html', {'artciles':page_data})
Django之数据库对象关系映射的更多相关文章
- 优酷项目之 ORM(数据库对象关系映射)代码重写
前言: 我们在操作数据库时候一般都是通过sql代码来操作mysql数据库中相关数据,这就需要懂得sql语句,那么怎么样才能在不懂sql语句的情况下通过我们所学的python代码来实现对mysql数据库 ...
- Python Web框架篇:Django Model ORM(对象关系映射)
一,基本操作 用于实现面向对象编程语言里不同类型系统的数据之间的转换,换言之,就是用面向对象的方式去操作数据库的创建表以及增删改查等操作. 1.增(create , save): from app01 ...
- web前端基础知识-(八)Django进阶之数据库对象关系映射
Django ORM基本配置 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接数据库,并编写数据访问层代码 业务逻辑层去 ...
- Python之路【第十九章】:Django 数据库对象关系映射
Django ORM基本配置 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接数据库,并编写数据访问层代码 业务逻辑层去 ...
- web框架-(四)Django进阶之数据库对象关系映射
Django ORM基本配置 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接数据库,并编写数据访问层代码 业务逻辑层去 ...
- Daject初探 - 一个开源关系型数据库对象关系映射(ORM)模型
Daject简介 Daject是用php写的一个关系型数据库抽象模型,通过该模型,可以在不写任何SQL或写很少的SQL就能执行大多数数据库查询操作.Daject具有面向对象,跨数据库的优点,通过数据库 ...
- Django 源码小剖: Django 对象关系映射(ORM)
引 从前面已经知道, 一个 request 的到来和一个对应 response 的返回的流程, 数据处理和数据库离不开. 我们也经常在 views.py 的函数定义中与数据库打交道. django O ...
- $Django setting.py配置 ,GET、POST深入理解,三件套,orm对象关系映射简介
1 django中app的概念: 大学:----------------- 项目 信息学院 ----------app01 物理学院-----------app02 ****强调***:创建的每一 ...
- Python学习---django之ORM语法[对象关系映射]180124
ORM语法[对象关系映射] ORM: 用面向对象的方式去操作数据库的创建表以及增删改查等操作. 优点:1 ORM使得我们的通用数据库交互变得简单易行,而且完全不用考虑该死的SQL语句.快速开发. 2 ...
随机推荐
- CentOS7下编译安装Python3.7.x【亲测有效】
所有操作都在root用户下操作 下载安装包 编译安装 建立软链接 验证 安装: 更新yum: yum update 安装Python依赖: yum install openssl-devel bzip ...
- Java 中抽象类与接口的区别
TypeScript 中的接口,有点类似抽象类的概念.Java 中抽象类属于包含属性与抽象行为,而接口通常只是抽象行为.抽象类可以实现模板模式. 参考 https://www.cnblogs.com/ ...
- 电脑视频下载王-Apowersoft Video Download Capture v6.3.6
Apowersoft Video Download Capture (视频下载王) 是由香港Apowersoft出品的一款集视频下载.视频转换.媒体播放及录屏等功能为一体的多功能视频下载工具,简便实用 ...
- javax.jms.JMSException: Failed to build body from content. Serializable class not available to broker. Reason: java.lang.ClassNotFoundException: Forbidden class com.javaliao.portal.model.TbLogVisit! T
javax.jms.jmsexception:无法从内容生成正文.可序列化类不可用于代理原因:java.lang.ClassNotFoundException: 禁止类com.javaliao.por ...
- vue2 单一事件中心管理组件通信
- group_concat 排序并取前三个
substring_index(group_concat(distinct num order by num desc),',',3)
- RMQ问题【模板】
概念 RMQ 是英文 Range Maximum/Minimum Query 的缩写,表示区间最大(最小)值. 解决 RMQ 问题的主要方法有两种,分别是 ST 表和线段树.本文主要讲 ST 表. S ...
- [ZJOI2009] 硬币游戏(找规律)
题目 洛谷传送门 题解 把1/21/21/2转化成0/10/10/1,所以直接可以异或. 对于长度为nnn的0/10/10/1数列,发现每变换2k(k>1)2^k(k>1)2k(k> ...
- nginx 作用,初认识
访问网站的dns域名解析 1.浏览器缓存之前解析的域名对应的ip 2.在本机host文件中配置的域名与ip对应的地址 3.最后都没映射,指向的域名的万网服务器,拿到真实域名对应的ip. nginx 作 ...
- 到spring官网创建第一个springboot工程
登录到spring的官网,直接生成一个,然后倒入本地工程就可以了. https://start.spring.io/ 点击创建的时候. 就等于下载了这个工程. 下载后,倒入到我们的maven工程可以直 ...