web前端基础知识-(八)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':'cc', #数据库名
'USER': 'root', #账号
'PASSWORD': '123456', #密码
'HOST': '192.168.28.129', #mysql数据库IP
'PORT': '3306', #对端端口
}
}
2、创建定义数据库表结构文件(对应app目录下的models.py文件)
生成一个简单的数据库表:
from django.db import models # Create your models here. class UserInfo(models.Model):
# 创建的表名名称为cmdb_userinfo
# 数据库默认创建id列 自增 主键
# 用户名列 字符串类型 字符长度
username = models.CharField(max_length=32) # max_length 字符长度
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',
'cmdb', # 系统会加载cmdb下的model.py文件
]
3、生成数据库表
执行下面命令:
python manage.py makemigrations
python manage.py migrate # 生成数据表
查看数据库表详情:
mysql> desc cmdb_userinfo;
+----------+-------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+----------+-------------+------+-----+---------+----------------+
| id | int(11) | NO | PRI | NULL | auto_increment |
| username | varchar(32) | NO | | NULL | |
| passwd | varchar(64) | NO | | NULL | |
+----------+-------------+------+-----+---------+----------------+
3 rows in set (0.01 sec)
注意:Django默认用的MysqlDB模块连接数据库,但在python3.x里面还没有这个模块,所有需要把连接数据库的模块改成pymsyql,修改project目录下的init.py文件
import pymysql
pymysql.install_as_MySQLdb()
4、数据库字段和字段参数
1、models.AutoField 自增列 = int(11)
如果没有的话,默认会生成一个名称为 id 的列,如果要显示的自定义一个自增列,必须将给列设置为主键 primary_key=True。
2、models.CharField 字符串字段
必须 max_length 参数
3、models.BooleanField 布尔类型=tinyint(1)
不能为空,Blank=True
4、models.ComaSeparatedIntegerField 用逗号分割的数字=varchar
继承CharField,所以必须 max_lenght 参数
5、models.DateField 日期类型 date
对于参数,auto_now = True 则每次更新都会更新这个时间;auto_now_add 则只是第一次创建添加,之后的更新不再改变。
6、models.DateTimeField 日期类型 datetime
同DateField的参数
7、models.Decimal 十进制小数类型 = decimal
必须指定整数位max_digits和小数位decimal_places
8、models.EmailField 字符串类型(正则表达式邮箱) =varchar
对字符串进行正则表达式
9、models.FloatField 浮点类型 = double
10、models.IntegerField 整形
11、models.BigIntegerField 长整形
integer_field_ranges = {
'SmallIntegerField': (-32768, 32767),
'IntegerField': (-2147483648, 2147483647),
'BigIntegerField': (-9223372036854775808, 9223372036854775807),
'PositiveSmallIntegerField': (0, 32767),
'PositiveIntegerField': (0, 2147483647),
}
12、!models.IPAddressField 字符串类型(ip4正则表达式)不再使用
13、models.GenericIPAddressField 字符串类型(ip4和ip6是可选的)
参数protocol可以是:both、ipv4、ipv6
验证时,会根据设置报错
14、models.NullBooleanField 允许为空的布尔类型
15、models.PositiveIntegerFiel 正Integer
16、models.PositiveSmallIntegerField 正smallInteger
17、models.SlugField 减号、下划线、字母、数字
18、models.SmallIntegerField 数字
数据库中的字段有:tinyint、smallint、int、bigint
19、models.TextField 字符串=longtext
20、models.TimeField 时间 HH:MM[:ss[.uuuuuu]]
21、models.URLField 字符串,地址正则表达式
22、models.BinaryField 二进制
23、models.ImageField 图片
24、models.FilePathField 文件
所有字段
null -> db是否可以为空
default -> 默认值
primary_key -> 主键
db_column -> 列名
db_index -> 索引
unique -> 唯一索引
unique_for_date ->
unique_for_month
unique_for_year
auto_now -> 创建时,自动生成时间
auto_now_add -> 更新时,自动更新为当前时间
choices -> django admin中显示下拉框,避免连表查询
blank -> django admin是否可以为空
verbose_name -> django admin显示字段中文
editable -> django admin是否可以被编辑
error_messages -> 错误信息欠
help_text -> django admin提示
validators -> django
form, 自定义错误信息(欠) 所有字段参数...
所有字段参数
Django 数据库操作
1. 增加数据
①常用方法
from cmdb import models
def ormadd(request):
models.UserInfo.objects.create(username="James",passwd="8888") return HttpResponse("ormadd")
②常用方法的另外一种写法
from cmdb import models
def ormadd(request):
dicts = {'username':"James",'passwd':"8888"}
models.UserInfo.objects.create(**dicts) return HttpResponse("ormadd")
③第二种方法
from cmdb import models
def ormadd(request):
obj = models.UserInfo(username='root',passwd='123')
obj.save() return HttpResponse("ormadd")
2. 删除数据
from cmdb import models
def ormdel(request):
models.UserInfo.objects.filter(id=3).delete() return HttpResponse("ormdel")
3. 更新数据
①第一种方式
from cmdb import models
def ormupdate(request):
models.User.objects.filter(id__gt=1).update(name='cc',age=18)
return HttpResponse("ormupdate")
②第二种方式
from cmdb import models
def ormupdate(request):
dic = {'name': 'cc', 'age': 19}
models.User.objects.filter(id__gt=1).update(**dic)
return HttpResponse("ormupdate")
4. 查询数据
①查询所有数据
from cmdb import models
def ormselect(request):
result = models.UserInfo.objects.all()
print(result) # QuerySet类型 列表里每个元素都是一个obj对象
# <QuerySet [<UserInfo: UserInfo object>, <UserInfo: UserInfo object>, <UserInfo: UserInfo object>]>
for row in result:
print(row.id,row.username,row.passwd)
# 1 James 8888
# 2 root 123
# 3 James 8888
return HttpResponse("ormselect")
②查找指定字段
from cmdb import models
def ormselect(request):
result = models.UserInfo.objects.filter(username='root') # filter()里可以用,分开 表示and
if result:
for row in result:
print(row.id,row.username,row.passwd)
# 2 root 123 return HttpResponse("ormselect")
③获取查询第一条数据和统计匹配个数
from cmdb import models
def ormselect(request):
obj = models.UserInfo.objects.filter(username='root').first() # 获取匹配的第一条字段
c = models.UserInfo.objects.filter(username='root').count() # 获取匹配的字段个数 return HttpResponse("ormselect")
④比较值查询以及多条件查询
models.User.objects.filter(id=1,name='root') # id等于1且name等于root
models.User.objects.filter(id__gt=1,name='root') # id大于1
models.User.objects.filter(id__lt=1) # id小于1
models.User.objects.filter(id__gte=1) # id大等于
models.User.objects.filter(id__lte=1) # id小等于
⑤指定查询数据库某个字段的所有数据
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<p>项目组(对象) v1</p>
<ul>
{% for row in v1 %}
<li>{{ row.id }}-{{ row.caption }}-{{ row.code }}</li>
{% endfor %}
</ul>
<p>项目组(字典) v2</p>
<ul>
{% for row in v2 %}
<li>{{ row.id }}-{{ row.caption }}</li>
{% endfor %}
</ul>
<p>项目组(元组) v3</p>
<ul>
{% for row in v3 %}
<li>{{ row.0 }}-{{ row.1 }}</li>
{% endfor %}
</ul>
</body>
</html>
business.html
获取表单数据:
def business(request):
v1 = models.Business.objects.all()# QuerySet ,内部元素都是对象
print("v1",v1)
# v1 < QuerySet[ < Business: Business object >, < Business: Business object >,
# < Business: Business object >, < Business: Business object >] > v2 = models.Business.objects.all().values("id","caption") # QuerySet ,内部元素都是字典
print("v2",v2)
# v2 < QuerySet[{'caption': '运维', 'id': 1}, {'caption': '开发', 'id': 2}, {'caption': '销售', 'id': 3},
# {'caption': '市场', 'id': 4}] > v3 = models.Business.objects.all().values_list("id","caption") # QuerySet ,内部元素都是元组
print("v3",v3)
# v3 < QuerySet[(1, '运维'), (2, '开发'), (3, '销售'), (4, '市场')] > return render(request,'business.html',{'v1':v1,'v2':v2,'v3':v3})
注:values和values_list后不指定字段的话,获取的是所有的字段数据
5. 单表外键关联
①数据库表结构
from django.db import models # Create your models here. class UserInfo(models.Model):
# 创建的表名名称为cmdb_userinfo
username = models.CharField(max_length=32)
passwd = models.CharField(max_length=64)
# 关联外键 生成字段名为user_group_id
user_group = models.ForeignKey("UserGroup",to_field='uid',default=1) #关联usergroup表的uid字段默认为1 class UserGroup(models.Model):
# 创建的表名名称为cmdb_usergroup
uid = models.AutoField(primary_key=True) #主键 自增
caption = models.CharField(max_length=32,unique=True) #唯一索引
ctime = models.DateField(auto_now_add=True,null=True) #创建时生成时间
uptime = models.DateField(auto_now=True,null=True) #更新是自动更新时间
②外键关联表操作
from cmdb import models
def ormadd(request):
models.UserInfo.objects.create(username="James",passwd="8888")
return HttpResponse("ormadd")
def ormgroup(request):
models.UserGroup.objects.create(caption="CEO")
return HttpResponse("ormgroup") def userinfo(request):
obj = models.UserInfo.objects.all().first()
# user_group_id 指定的是userinfo表里存的id号
print(obj.id,obj.username,obj.user_group_id)
# 1 James 1 # user_group 指定的是用户组obj对象
print(obj.user_group.uid,obj.user_group.caption)
# 1 CEO
return HttpResponse("userinfo")
③跨表查询,神奇的双下划綫__
class Business(models.Model):
# id
caption = models.CharField(max_length=32)
code = models.CharField(max_length=32,null=True,default="SA") class Host(models.Model):
nid = models.AutoField(primary_key=True)
hostname = models.CharField(max_length=32,db_index=True)
ip = models.GenericIPAddressField(protocol="ipv4",db_index=True)
port = models.IntegerField()
b = models.ForeignKey(to="Business", to_field='id')
models.py
获取表单数据:
def home(request):
# b__capiton 通过双下划线直接跨表获取对应caption的数据
v2 = models.Host.objects.filter(nid__gt=0).values('nid','hostname','b_id','b__caption')
# < QuerySet[{'nid': 2, 'b__caption': '开发', 'hostname': '系统1', 'b_id': 2},
# {'nid': 3, 'b__caption': '销售','hostname': '系统2', 'b_id': 3}> return render(request, 'home.html')
Django数据库多对多操作
1. 自定义创建关系表
class Host(models.Model):
nid = models.AutoField(primary_key=True)
hostname = models.CharField(max_length=32,db_index=True)
ip = models.GenericIPAddressField(protocol="ipv4",db_index=True)
port = models.IntegerField()
b = models.ForeignKey(to="Business", to_field='id') # 这里先忽略 class Applications(models.Model):
name = models.CharField(max_length=32) class HostToAPP(models.Model):
# 可以自定义多种字段
hobj = models.ForeignKey(to='Host',to_field='nid')
aobj = models.ForeignKey(to='Applications',to_field='id')
创建数据:
HostToAPP.objects.create(hobj_id=1,aobj_id=2)
2. 自动创建关系表
class Host(models.Model):
nid = models.AutoField(primary_key=True)
hostname = models.CharField(max_length=32,db_index=True)
ip = models.GenericIPAddressField(protocol="ipv4",db_index=True)
port = models.IntegerField()
b = models.ForeignKey(to="Business", to_field='id') # 这里先忽略 class Applications(models.Model):
name = models.CharField(max_length=32) r = models.ManyToManyField("Host")
自动生成第三张关联表cmdb_applications_r;下面我们就来看看针对这种情况下,如何对表进行操作
①增加
# 第三张表增加
obj = models.Application.objects.get(id=1) obj.r.add(1) # 创建id=1,nid=1
obj.r.add(2) # 创建id=1,nid=2
obj.r.add(2, 3, 4) # 创建id=1,nid=2、id=1,nid=3 、id=1,nid=4
obj.r.add(*[1, 2, 3, 4]) # 同上
②删除
# 第三张表删除
obj = models.Application.objects.get(id=1) obj.r.remove(1) # 删除id=1,nid=1
obj.r.remove(2,4) # 删除id=1,nid=2、id=1,nid=4
obj.r.remove(*[1,2,3]) # 批量删 obj.r.clear() # 所有id=1的关系删除
③更新
# 第三张表更新
obj = models.Application.objects.get(id=1) obj.r.set([3,5,7]) # 清空原来的id=1关系,创建id=1,nid=3;id=1,nid=5;id=3,nid=7
# 等同于执行obj.r.clear()再执行obj.r.add(*[3,5,7])
④查询
# 第三张表查询
obj = models.Application.objects.get(id=1) # 所有相关的主机对象“列表” QuerySet
obj.r.all() # 所有对应的Host对象
Django练习
上面学习了很多知识,然并卵,咱们还是结合具体的实例看下吧~!
web前端基础知识-(八)Django进阶之数据库对象关系映射的更多相关文章
- web框架-(四)Django进阶之数据库对象关系映射
Django ORM基本配置 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接数据库,并编写数据访问层代码 业务逻辑层去 ...
- web前端基础知识及快速入门指南
web前端基础知识及快速入门指南 做前端开发有几个月了,虽然说是几个月,但是中间断断续续的上课.考试以及其它杂七杂八的事情,到现在居然一直感觉自己虽然很多前端的知识很眼熟,却也感觉自己貌似也知识在门口 ...
- Python学习---django之ORM语法[对象关系映射]180124
ORM语法[对象关系映射] ORM: 用面向对象的方式去操作数据库的创建表以及增删改查等操作. 优点:1 ORM使得我们的通用数据库交互变得简单易行,而且完全不用考虑该死的SQL语句.快速开发. 2 ...
- [原创]java WEB学习笔记81:Hibernate学习之路--- 对象关系映射文件(.hbm.xml):hibernate-mapping 节点,class节点,id节点(主键生成策略),property节点,在hibernate 中 java类型 与sql类型之间的对应关系,Java 时间和日期类型的映射,Java 大对象类型 的 映射 (了解),映射组成关系
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- web前端基础知识学习网站推介
内容:一.基础知识及学习资料1. HTML入门学习:http://www.w3school.com.cn/html/index.aspHTML5 入门学习:http://www.w3school.co ...
- web前端基础知识总结
上个寒假总结的web前端的一些知识点给大家分享一下 1.<html>和</html> 标签限定了文档的开始和结束点. 属性: (1) dir: 文本的显示方向,默认是从左向右 ...
- web前端基础知识-(六)web框架
一.web框架本质 众所周知,对于所有的Web应用,本质上其实就是一个socket服务端,用户的浏览器其实就是一个socket客户端. #!/usr/bin/env python #coding:ut ...
- web前端基础知识-(七)Django进阶
通过上节课的学习,我们已经对Django有了简单的了解,现在来深入了解下~ 1. 路由系统 1.1 单一路由对应 url(r'^index$', views.index), 1.2 基于正则的路由 u ...
- web前端基础知识 - Django进阶
1. 路由系统 1.1 单一路由对应 url(r'^index$', views.index), 1.2 基于正则的路由 url(r'^index/(\d*)', views.index), url( ...
随机推荐
- MVC下压缩输入的HTML内容
在MVC下如何压缩输出的HTML代码,替换HTML代码中的空白,换行符等字符? 1.首先要了解MVC是如何输出HTML代码到客户端的,先了解下Controller这个类,里面有很多方法,我们需要的主要 ...
- Java实现上传下载
一.上传 二.下载 import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.Fi ...
- 终于将rsync-3.1.2配置成功,之外还挖掘了一些新的用法
1.为什么要用rsync: 有两台主机,开始准备做HA,考虑到工作量的问题,最终决定将重要文件进行同步即可. 找了一些同步的工具,rsync得到一致好评,速度快,消耗小等等. 2.接着找资料,最后选用 ...
- sql语句with as 和with(nolock)
当with和as一起用时,表示定义一个SQL字句 例: with sonword as ( select * from person ) select * from student where n ...
- .htaccess添加Header set Cache-Control报错500
在优化网站开启站点的图片缓存时,需要在.htaccess文件中加入: #文件缓存时间配置10分钟 <FilesMatch ".(flv|gif|jpg|jpeg|png|ico|swf ...
- 前端开发必备!Emmet使用手册
介绍 Emmet (前身为 Zen Coding) 是一个能大幅度提高前端开发效率的一个工具: 基本上,大多数的文本编辑器都会允许你存储和重用一些代码块,我们称之为"片段".虽然片 ...
- [译]Thinking in React
编者按 使用React的思想来构建应用对我在实际项目中以及帮助他人解决实际问题时起到了很大作用,所以我翻译此文来向那些正在或即将陷入React或React-Native深坑的同胞们表示慰问.网上已经有 ...
- C语言调试过程中duplicate symbol错误分析
说明:在我们调试C语言的过程中,经常会遇到duplicate symbol错误(在Mac平台下利用Xcode集成开发环境).如下图: 一.简单分析一下C语言程序的开发步骤. 由上图我们可以看出C语言由 ...
- java基础算法之插入排序
一.插入排序介绍 插入排序(Insertion Sort)是一种简单直观的排序算法.它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入.插入排序在实现上,通 ...
- Quartz框架
Quartz框架 Quartz 是个开源的作业调度框架,为在 Java 应用程序中进行作业调度提供了简单却强大的机制.Quartz 允许开发人员根据时间间隔(或天)来调度作业.它实现了作业和触发器的多 ...