二、models

一、数据库基本操作(添、删、改、查)

1.创建数据库表

====models.py====
from django.db import models

class User(models.Model):
      username = models.CharField(max_length=32)
      password  = models.CharField(max_length=64)
     
class Meta:
# 数据库中生成的表明,默认app名称+下划线+类名
db_table = 'user' 用user命名表名

index_together = [
("name",'pwd'),
]

#最左前缀的模式:
# select * from where name = 'xx'
# select * from where name = 'xx' and email = 'xx'
# select * from where email = 'xx' #无法命中索引

unique_together = (("driver","restaurant"),)
verbose_name = "上课记录"
verbose_name_plural=""上课记录"" ##创建数据库### python manage.py makemigrations python manage.py migrate

2.连接mysql

在工程文件目录中的__init__.py文件,添加如下内容:
import pymysql

pymysql.install_as_MySQLdb()

3.数据操作

===views.py====
from app01 import models
from django 

#增加数据
def orm(request):
     #方法一:
      models.UserInfo.objects.create(
                  username='root',
                  password='123'
            )     
      #方法二:
       obj = models.UserInfo(username='root',password='123')  
       obj.save()
      #第一种形式的另外一种方法
       dic = {'username':'eric','password':'111'}
       models.UserInfo.objects.create(**dic)
     
       return request.HttpResponse('OK')
#查询数据
     1.查看所有
      result = models.UserInfo.objects.all()   查看所有数据
      返回值:
      <Queryset [<UserInfo:UserInfo object>,<UserInfo:UserInfo object>,<UserInfo:UserInfo object>]>  返回值为Queryset类型,此类型是django提供,实质上是一个列表,列表中的每一个元素都是一个userinfo对象:
      #[obj(id,username,passowrd),obj(id,username,passowrd),obj(id,username,passowrd)]
     #获取结果的方式:
      for row in result:
           print (row.id,row.name,row.password)
    2.查看某些数据:
    result = models.UserInfo.objects.filter(username='root')
    得到结果的类型同上:

#删除数据
    models.UserInfo.objects.filter(id=4).delete()  #删除id等于4的用户


#更新数据
     models.UserInfo.objects.all().update(password=6660)
     models.UserInfo.objects.filter(id=3).update(password=6660)
   

4.用户登录验证实例:

===views.py===
def login(request):
      if request.method=="GET":
           return render(request,'login.html')
      elif  request.method=='POST':
           u = request.POST.get('user')
           p = request.POST.get('pwd')
           obj = models.UserInfo.objects.filter(username=u,password=p).first()
           if obj:
                return render(request,'/index/')
            else:

                return render(request,'login.html')
 

===login.html===
===index.html===

 二、Django字段类型

一)字段

1.数字类

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):
        - 小整数 -3276832767

PositiveSmallIntegerField(PositiveIntegerRelDbTypeMixin, IntegerField)
        - 正小整数 032767
IntegerField(Field)
        - 整数列(有符号的) -21474836482147483647

PositiveIntegerField(PositiveIntegerRelDbTypeMixin, IntegerField)
        - 正整数 02147483647

BigIntegerField(IntegerField):
        - 长整型(有符号的) -92233720368547758089223372036854775807
自定义无符号整数字段:
        class UnsignedIntegerField(models.IntegerField):
            def db_type(self, connection):
                return 'integer UNSIGNED'

        PS: 返回值为字段在数据库中的属性,Django字段默认的值为:
            '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)',


    FloatField(Field)
        - 浮点型

    DecimalField(Field)
        - 10进制小数
        - 参数:
            max_digits,小数总长度
            decimal_places,小数位长度

    BinaryField(Field)
        - 二进制类型
 

2.二进制型:

    BooleanField(Field)
        - 布尔值类型

    NullBooleanField(Field):
        - 可以为空的布尔值

3.字符串型

    CharField(Field)
        - 字符类型
        - 必须提供max_length参数, max_length表示字符长度

4.文本型

    TextField(Field)
        - 文本类型


5.邮件地址

    EmailField(CharField):
        - 字符串类型,Django Admin以及ModelForm中提供验证机制

6.IP地址:

    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"

7.日期类型:

   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类型

 

8.URL地址:

 



    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   上传图片的宽度保存的数据库字段名(字符串)
 

二)字段参数:

null                数据库中字段是否可以为空
db_column           数据库中字段的列名
db_tablespace
default             数据库中字段的默认值
primary_key         数据库中字段是否为主键
db_index            数据库中字段是否可以建立索引
unique              数据库中字段是否可以建立唯一索引
unique_for_date     数据库中字段【日期】部分是否可以建立唯一索引
unique_for_month    数据库中字段【月】部分是否可以建立唯一索引
unique_for_year 数据库中字段【年】部分是否可以建立唯一索引
auto_now 创建时,自动生成
auto_now_add 更新时,自动更新为当前时间 #obj = UserGroup.objects.filter(id=1).update(capiton='CEO')对应的时间字段(auto_now_add)不会进行更新。
#obj=UserGroup.objects.filter(id=1).first()
#obj.caption = "CEO"
#obj.save() #这种方式对应的时间才会进行更新 verbose_name Admin中显示的字段名称,标签名称 blank Admin中是否允许用户输入为空 editable Admin中是否可以编辑 help_text Admin中该字段的提示信息 choices Admin中显示选择框的内容,用不变动的数据放在内存中从而避免跨表操作 如:gf
= models.IntegerField(choices=[(0, '何穗'),(1, '大表姐'),],default=1)
class UserInfo(models.Model):
                username = models.CharField(max_length=32)
                password = models.CharField(max_length=64)
                
user_type_choices = (
                    (1,'超级用户')
                    (2,'管理员')
                    (3,'普通用户')
                )
               user_type_id = models.IntegerField(choices=user_type_choices,default=1)
IntegerField 指明数据库中存储的是数字,default表示默认即为"超级用户" error_messages 自定义错误信息(字典类型),从而定制想要显示的错误信息; 字典健:
null, blank, invalid, invalid_choice, unique, and unique_for_date 如:{'null': "不能为空.", 'invalid': '格式错误'} validators 自定义错误验证(列表类型),从而定制想要的验证规则 from django.core.validators import RegexValidator from django.core.validators import EmailValidator,URLValidator,DecimalValidator,\ MaxLengthValidator,MinLengthValidator,MaxValueValidator,MinValueValidator 如: test = models.CharField( max_length=32, error_messages={ 'c1': '优先错信息1', 'c2': '优先错信息2', 'c3': '优先错信息3', }, validators=[ RegexValidator(regex='root_\d+', message='错误了', code='c1'), RegexValidator(regex='root_112233\d+', message='又错误了', code='c2'), EmailValidator(message='又错误了', code='c3'), ] )

三、表之间的关系

一)一对多跨表查询:

=====models.py=======

class Foo(models.Model):
name = models.CharField(max_length=1) class Host(models.Model): nid = models.AutoField(primary_key=True) hostname = models.CharField(max_length=32) ip = models.GenericIPAddressField(db_index=True) port = models.IntegerField() b = models.ForeignKey(to="Business",to_field='id')

ForeignKey(ForeignObject) #ForeignObject(RelatedField)
to,      #要进行关联的表明
to_field=None #要关联的表中的字段名称
on_delete =None #当删除关联表中的数据时,当前表与其关联的行的行为
                        - models.CASCADE,删除关联数据,与之关联也删除
                                        - models.DO_NOTHING,删除关联数据,引发错误IntegrityError
                                        - models.PROTECT,删除关联数据,引发错误ProtectedError
                                        - models.SET_NULL,删除关联数据,与之关联的值设置为null(前提FK字段需要设置为可空)
                                        - models.SET_DEFAULT,删除关联数据,与之关联的值设置为默认值(前提FK字段需要设置默认值)
                                        - models.SET,删除关联数据,
                              a. 与之关联的值设置为指定值,设置:models.SET(值)
                                                  b. 与之关联的值设置为可执行对象的返回值,设置:models.SET(可执行对象)
            def func():
                 return 10

            class MyModel(models.Model):
                  user = models.ForeignKey(
                  to="User",
                  to_field="id"
                  on_delete=models.SET(func),)




class Business(models.Model) caption = models.CharField(max_length=32) fk = models.ForeignKey('Foo') #1 运维 #2 开发 #3 市场 #4 测试 ====views.py======= from django.shortcuts import render from app01 import models def business(request):
#1.单表,获取数据类型
#列表中为对象 v1
= models.Business.objects.all()
#QuerySet
#[obj(id,caption,code),obj(id,caption,code),obj(id,cacpiton,code)]

models.xx.objects.filter(id=1,name='root')
models.xx.objects.filter(id__gt=1)
models.xx.objects.filter(id__lt=5)
models.xx.objects.filter(id__get=1)
models.xx.objects.filter(id__lte=5) #列表中为对象

#列表中为字典
v2 = models.Business.objects.all().values('id','caption')
#QuerySet
#[{'id':1,'caption':'运维部'},{'id':3,'capiton':'市场'},{'id':2,'caption':'开发'},{'id':4,'caption':'测试'}]

v3 = models.Business.objects.aall().values_list('id','caption')
#QuerySet
#[(1,运维),(2,开发),(3,市场),(4,测试)]

#############以上均为QuerySet类型###################

获取到的是一个对象,不是QuerySet类型,如果没有获取数据,则直接抛出异常
models.Business.objects.get(id=1)

以下获取的单个对象,如果不存在则会返回None,不会出现报错
models.Business.objects.filter(id=1).first() 获取QuerySet对象
return render(request,'business.html',{'v1':v1,'v2':v2,'v3':v3})
    #2.跨表获取数据查询
a.通过.进行跨表查询
    def Host(models.Model):
v1 = models.Host.objects.filter(nid__gt=0)
for row in v1:
print (row.nid,row.hostname,row.ip,row.port,row.b_id,row.b.caption,row.b.code,row.b.id,sep="\t")

print (row.nid,row.hostname,row.ip,row.port,row.b_id,row.b.caption,row.b.code,row.b.id,sep="\t")
     v1 = models.Host.objects.filter(nid__gt=0)  
for row in v1:
print (row.b.fk.name)

v = models.Host.objects.filter(nid__gt=0) #v1QuerySet类型的列表,所以也可以用下面的方式

       v[0].b.caption     ---->通过.进行跨表查询

       #注,以上均为已经获取对象,然后从对象中取值,因此可以用.获取数据

 

      b.通过__(双下划线)进行块表查询

    v2 = models.Host.objects.filter(nid__gt=0).values('nid','hostname','b_id','b.caption') ,此种方式不能通过.进行跨表查询,b.caption会引起报错
需要写成 b__caption,此时b__caption是一个字符串
v2 = models.Host.objects.filter(nid__gt=0).values('nid','hostname','b_id','b__caption')
for row in v2:
print(row['nid'],row['hostname'],row['b_id'],row['b__caption'])

#注,当通过models.xx.objects这种方式,后面所有数据字段均为字符串(如b.caption),因此只能通过b__caption获取数据。这个是django中获取数据的一种方约定


前端的获取方式:
{% for row in v2 %}
<tr hide="{{ row.nid }}" bid="{{ row.b_id }}">
<td>{{ row.hostname }}</td>
<td>{{ row.b__caption}}
</tr>
{% endfor %}
      v3 = models.Host.objects.filter(nid__gt=0).values_list('nid','hostname','b_id','b__caption')
return render(request,'host.html',{'v1'})

前端的获取方式:
      {% for row in v3 %}
<tr hide="{{ row.0 }}" bid="{{ row.2 }}">
<td>{{ row.1 }}</td>
<td>{{ row.3}}
</tr>
{% endfor %}
===business.html==== 
<h1>业务线列表(对象调用)</h1>
<ul>
{
% for row in v1 %}
<li>{{ row.id }}-{{ row.caption }}-{{ row.code }}</li>
{
%endfor%}
</ul>

    <h1>业务线列表(字典调用)</h1>
    <ul>
          {% for row in v2 %}
                   <li>{{ row.id }}-{{ row.caption }}</li>
           {%endfor%}
    </ul>  

<h1>业务线列表(元祖调用)</h1>
<ul>
{% for row in v3 %}
<li>{{ row.0 }}-{{ row.1 }}</li>
{% endfor %}
</ul>

====host.html======
<table border='1'>
<thead>
<tr>
<th>主机名</th>
<th>IP</th>
<th>端口</th>
<th>业务线名称</th>
</tr>
</thead>
<tbody>
{% for row in v1 %}
<tr host-id="{{ row.nid }} bid="{{ row.b.id }}""> #主机id和业务线id
<td>{{ forloop.counter }}</td>
# forloop.counter0
# forloop.revercounter
# forloop.revercounter0
# forloop.last #是否是左后一个,最后一个返回True,
             # forloop.first
#forloop.parentloop
<td>{{ row.hostname }}</td>
<td>{{ row.ip }}</td>
<td>{{ row.port }}</td>
<td>{{ row.b.capiton}}</td>
</tr>
{% endfor%}
</tbody>
</table>


 

二)多对多:

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 Application(models.Mode):
        name = models.CharField(max_length=32)

class  HostToApp(models.Model):
        hobj = models.ForeignKey(to='Host',to_field='nid')
        aobj = models.ForeignKey(to='Application',to_field='id')
        status = models.CharField(max_length=32)

# HostToApp.objects.create(hobj_id=1,aobj=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 Application(models.Mode): name = models.CharField(max_length=32) r = models.ManyToManyField("Host")
       #   1    WEB
# 2 DB
# 3 Cache
# 4 Proxy


自动生成application_r关系表,表的内容:
id application_id host_id


=============================================================================== #无法直接对第三张表进行操作
obj = Application.objects.get(id=1)
obj.name
#添加
obj.r.add(1) 增加一个application id=1,host nid等于1的
obj.r.add(2)
增加一个application id=1,host nid等于2的
obj.r.add(3,4,5)
obj.r.add(*[1,2,3,4]) 增加多个关系,增加application id=1,host nid等于1,2,3,4

#删除
obj.r.remove(1) 删除application id=1 host nid 等于1
obj.r.remove(2,3,4)
obj.r.remove(*[1,2,3,4]) 删除application id=1 host nid等于1,2,3,4,

#清除
obj.r.clear() 清除application id为1的所有行

#更新
obj.r.set([3,5,7]) #不加*, 数据表中只会存在1,3 | 1,5 | 1,7 行,其它行都被删除

#获取
obj.r.all() 获取application id为1的所有行,拿到的是QuerySet对象


=====views.py======
def app(request):
app_list = models.Application.objects.all()
for row in app_list:
print (row.name,row.r.all())

====app.html=======
<table border='1'>
<tr>
<td>应用名称</td>
   <td>应用主机列表</td>
</tr>
<tbody>
{% for app in app_list %}
<tr>
<td>{{ app.name }}</td>
<td>
            #{{ app.r.all }} 主机列表
            {% for host in app.r.all %}
<span> {{ host.hostname }} </span>
{% endfor %}
         </td>
</tr>
{% endfor %}
</tbody>
</table>

 

posted @ 2017-03-13 20:10  jidi_78  阅读(79)  评论(0)    收藏  举报