Django之MTV实战(2)

[toc]

Hello, 各位,我回来了,大家别以为我消失了,我还是在的...

最近忙于家里重要事情,不能定期及时更新,请包含...

忙里挑一,我还是在后台默默的码了几篇文章,前提要保证下质量,才能发出来,哈哈!不然...嘿嘿

大家搬好小板凳了,前方的真的高能,文章篇幅有点多,一步一步来...

跟着我走,简单学起来...

1. 回顾知识

上一篇文章已经教会了大家怎么安装Django和简单的配置,相信大家应该早就学会了,那么我们在回忆一下吧,懂的同学可跳过这章节。

1.1 新增工程

django-admin startproject <自定义工程名称>

(py369) [python@localhost Python]$ django-admin startproject devops 

1.2 创建新的APP

python manage.py startapp <自定义APP名称>

(py369) [python@localhost devops]$ python manage.py startapp hello

1.3 注册APP

devops->settings.y里面t添加:

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    # 第一种方式
    'hello.apps.HelloConfig',
    # 第二种方式,直接写hello也行
    'hello',
]

1.4 编写URL和VIEW

在devops下的主路由urls.py

from django.contrib import admin
from django.urls import path,include
from views import index

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', index.index),
    # 引导到hello下的路由URL(也叫子路由)
    path('hello/', include('hello.urls'))
]

在hello下的子路由urls.py

from django.urls import path
from hello import view

app_name = 'hello'
urlpatterns = [
    # 普通url参数
    path('', view.index, name='index'),

hello下的view.py代码:

from django.http import HttpResponse

def index(request):
    return HttpResponse('hello django')

1.5 验证结果如下:

Django之MTV实战(2)_第1张图片

2. 基本概念

2.1 专业术语

MTV简写:

  • M:model,这个是对应数据库的,简单理解就是对应数据库的表。
  • T:template,这个对应的是HTML模板,前端渲染用的。
  • V:view,这个对应的是后台python执行脚本了。

通俗的一句话:用户发送http请求,匹配url后执行view脚本返回模板template,用户看到了网页的展示效果(渲染)

2.2 MTV之视图

2.2.1 request对象

Django之MTV实战(2)_第2张图片

2.2.2 Respone对象

Django之MTV实战(2)_第3张图片

下面详细介绍下...

2.2.3 GET请求

  • GET请求,不带参数

    网页输入这样的格式,是不带参数

    https://192.168.8.130:8888/hello

    备注:如上面演示的就是不带参数。

  • GET请求,?+参数

    比较常用的方式?+参数

    在浏览器输入如下地址:

    http://192.168.8.130:8888/hello/?year=2020&month=09&day=02

    说明: 参数:year month day

    网址匹配到路由hello/url.py 的配置规则

    from django.urls import path
    from hello import view
    
    app_name = 'hello'
    urlpatterns = [
        # 普通参数
        path('', view.index, name='index'),
    ]

    后台视图hello/view.py代码配置如下:

    from django.http import HttpResponse
    
    def index(request):
        print(request.GET)
        return HttpResponse("year is {}, month is {}, day is {}.".format(year, month, day))

    后台打印输出的结果如下:
    备注: 是一个QueryDict对象。

    从上面已经接收到用户的信息了,就可以获取相应的参数了,hello/view后台脚本更新如下:

      from django.http import HttpResponse
      
      def index(request):
          #第一个参数是获取QueryDict的year
          #第二参数是默认值,表示拿不到数据,用缺省值
          year = request.GET.get('year', '2030')                                 
          month = request.GET.get('month', 'Sep')
          day = request.GET.get('day', '8')
          return HttpResponse("year is {}, month is {}, day is {}.".format(year, month, day))

    网页请求带参数返回的结果如下:

    网页请求不带参数返回的结果如下:

    Django之MTV实战(2)_第4张图片

  • GET请求,位置参数

    不推荐使用,位置要一一对应入座

    网址匹配到路由hello/url.py配置规则

    from django.urls import re_path
    from hello import view
    
    app_name = 'hello'
    urlpatterns = [
        # 位置参数
        # [0-9]表示数字0-9,{4}表示取4位数字
        re_path('([0-9]{4})/([0-9]{2})/([0-9]{2})/', view.index, name='index'),
    ]

    后台视图hello/view.py脚本配置如下:

    def index(request, year, month, day):
        return HttpResponse("year is {}, month is {}, day is {}.".format(year, month, day))

    网页输入如下地址,请求返回的结果如下:

    Django之MTV实战(2)_第5张图片

  • GET请求,关键字参数

    说明:强烈推荐,优雅的方式.

    在浏览器输入如下地址:

    http://192.168.8.130:8888/2020/09/02

    路由视图hello/url.py配置规则

    from django.urls import re_path
    from hello import view
    
    app_name = 'hello'
    urlpatterns = [
        # 关键字参数,(?<参数名>参数类型)
        re_path('(?P[0-9]{4})/(?P[0-9]{2})/(?P[0-9]{2})', view.index, name='index'),
    ]

    后台视图hello/view.py脚本配置如下:

    from django.http import HttpResponse
    
    def index(request, **kwargs):
        # 输出结果:{'year': '2020', 'month': '09', 'day': '02'}
        print(kwargs)  
        year = kwargs.get('year')
        month = kwargs.get('month')
        day = kwargs.get('day')
        return HttpResponse("year is {}, month is {}, day is {}.".format(year, month, day))

    还可以换成另外一种写法,更加灵活,但是用的也不是很多:

    from django.http import HttpResponse
    
    # 不用考虑到函数参数的位置
    def index(request, day, month, year):
        return HttpResponse("year is {}, month is {}, day is {}.".format(year, month, day))

2.2.4 POST请求

在devops/setting.py里把csrf关闭,不然会运行报错:

MIDDLEWARE = [
    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    # 默认开启防止中间人CSRF攻击,前期先注释掉
    # 'django.middleware.csrf.CsrfViewMiddleware',  
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

网址匹配到路由hello/urls.py配置规则

from django.urls import path
from hello import view

app_name = 'hello'
urlpatterns = [
    path('', view.index, name='index'),
]

后台视图hello/view.py脚本配置如下:

from django.http import HttpResponse, QueryDict

def index(request):
    if request.method == "POST":
        # POST方法
        print(request.method) 
        # body是字节编码,b'year=2020&month=09&day=13'
        print(request.body)  
        # 转换为字典{'year': '2020', 'month': '09', 'day': '13'}
        print(QueryDict(request.body).dict())
        # 
        print(request.POST)  
        data = request.POST
        year = data.get('year', '2030')
        month = data.get('month', '9')
        day = data.get('day', '8')

        return HttpResponse("year is {}, month is {}, day is {}.".format(year, month, day))

模拟触发POST流量:

[root@localhost ~]# curl -X POST http://192.168.8.130:8888/hello/ -d 'year=2020&month=09&day=13'
year is 2030, month is 9, day is 13.

看看我们后台接收哪些信息:

Django之MTV实战(2)_第6张图片

2.2.5 QueryDict介绍

在httprequest对象中,GET和POST属性是django.http.QueryDict的实例,它是一个自定义的类似字典的类,用来处理同一个键带多个值。无论使用GET,POST方式,他们最终都是通过QueryDict方法对传入的参数进行处理。

3. MTV之模板

3.1 模板继承

3.1.1 常规手段

  • 创建模板templates目录及子目录hello

    mkdir -p devops/templates/hello

    备注:每一个APP对应一个目录。

  • 路由视图hello/urls.py配置规则

    from django.urls import path
    from hello import view
    
    app_name = 'hello'
    urlpatterns = [
        path('list/', view.list, name='list'),
    ]
  • 后台视图hello/view.py配置

    from django.shortcuts import render
    
    def list(request):
        users = [
            {'username':'test01', 'age':18, 'hobby':'python'},
            {'username':'test02', 'age':18, 'hobby':'java'},
            {'username':'test01', 'age':18, 'hobby':'C'},
        ]
        return render(request, 'hello/list.html', {'users':users})

    说明:本次练习,还没涉及到数据库,所以先本地创建数据。

  • 新建模板

    templates/hello/list.html配置

      
      
      
          
          点滴技术
      
      
      

    用户列表

    {% for user in users %} {% endfor %}
    username age hobby
    {{ user.username }} {{ user.age }} {{ user.hobby }}

    版权所有©点滴技术

  • 网页输入地址后,效果图:
    Django之MTV实战(2)_第7张图片

3.1.2 模板继承

  • 定义母板
    devops/templates目录下新增一个base.html母板。

      
      
      
      
        
          {% block title %}NetDevOps{% endblock title %}
        
      
    
      
      
      {% block body %}这是body的内容{% endblock body %}
    
      
      

    版权所有©点滴技术

  • 子页面继承

      
      {% extends "base.html" %}
      
      {% block title %} 用户的列表 {% endblock %}
    
      
      {% block body %}
      
              {% for user in users %}
              
              {% endfor %}
          
    username age hobby
    {{ user.username }} {{ user.age }} {{ user.hobby }}
    {% endblock%}

    备注:公共部分代码就不用写出来了,减少了代码冗余。

  • 视图hello/view.py配置

    from django.shortcuts import render
    
    def userlist(request):
        users = [
            {'username':'test01', 'age':18, 'hobby':'python'},
            {'username':'test02', 'age':18, 'hobby':'java'},
            {'username':'test03', 'age':18, 'hobby':'C'},
        ]
        return render(request, 'hello/userlist.html', {'users':users})
  • 效果图:

    Django之MTV实战(2)_第8张图片

4. Template模板过滤器

4.1 Django自带常用过滤器

  • 传入参数的长度

    {% if messages|length >= 3 %}
     The Messages is too long.
    {% else %}
     The messages is too short.
    {% endif %}
  • default:缺省值

    {{ messages|default:"nothing" }}

    备注:如果传入的值为false,则使用缺省值。

  • first/last

    {{ messages|first }}
    {{ messages|last }}

    备注:显示列表第一个或最后一个元素。

  • join
    说明:将列表转为字符串。

    {{ value|join:"-" }}
  • length
    说明:判断长度,返回布尔值

    {{ messages|length}}
    {{ messages|length_is:"4"}}
  • static
    说明:加载本地图片、css、js样式等资源,通常使用CDN方式。

    # 方法1:
    {% load static %}
    Hi!
    
    # 方法2:
    {% load static %}
    {% static "images/favicon.png" as myphoto %}
    
  • date
    说明:时间格式化,返回年-月-日 时-分-秒

    {{ messages|date:"Y/m/d" }}{{ messages|date:"H:i:s" }}
  • safe
    说明:缺省情况下,django会对HTML等标签进行自动转义,如果要关闭自动转义,可通过过滤器"|safe"的方式申明不用转义。

    value = " 百度链接 "
    {{ value|safe }}
  • csrf_token
    说明:用于跨站请求伪造保护

       
    {% csrf_token %} # 有了这个POST请求才能正常运行

  • slice
    说明:切片

    {{ messages|slice:":2"}}
    

4.2 自定义模板标签和过滤器

  • 定义标签
    创建目录及文件:hello/templatetags/mytag.py

    from django import template
    
    register = template.Library()
    
    @register.filter
    def test(x, y):
        return int(x)*2 + int(y)qq
  • 模板视图

      
      {% extends "base.html" %}
      {% block title %}模板标签{% endblock %}
    
      
      {% block body %}
    
      
      {% load mytag %}
      

    {{ "2"|test:"1" }}

    {% endblock%}

5. 模型Model基础

5.1 模型概念

简单理解:模型对应数据库中的表,模型中的一个类对应数据库一张表;

5.1.1 常用字段类型

  • 字符串:CharFieLd

    from django.db import models
    class User():
        username = models.CharField(max_length=20)
  • 整数:IntegerField

    int_field = models.IntegerField()
  • 浮点数:FloatField

    float_field = models.FloatField()
  • 自增字段:AutoField

    id_field = models.AutoField(primary_key=True)
  • 文本框:TextField

    text_field = models.TextField()
  • 邮箱:EmailField
    说明:用于检查邮箱的合法性。

    mail_field = models.EmailField()
  • 日期:DateField

    说明:auto_now是被保存时,将时间设置为当前时间,通常表示last-modified, auto_now_add是首次被创建时,设置为当前时间,通常表示创建时间。

    date = models.DateField()
  • 文件上传:Filefield
    说明:upload_to必选参数,指文件的上传存放路径。

    upload_file = models.FileField(upload_to='/usr/tmp/test')

5.1.2 常用字段参数

  • null
    如果null=True将再数据库存放一个空值NULL,缺省为Flase。
    该字段是可以在数据中存放null值。
  • blank
    如果blank=True,则允许该字段为空白,缺省是False,不允许为空。
    该字段是表单验证是否允许为空或不为空的。
  • unique
    如果unique=True,表示该字段在整个表单中是唯一的,不重复的。
  • primary_key
    如果primary_key=True, 表示该字段在数据库中是主键。
  • default = ''
    用于定义缺省值。
  • verbose_name
    ForeignKeyManyToManyField、和OneToOneField的备注信息需要用到这个。

6. 建模及同步

6.1 设计一个简单的模型

hello\\models.py

#!/usr/bin/env python3
#-*- coding:UTF-8 -*-

from django.db import models

class Devices(models.Model):
    device_name = models.CharField(max_length=32, help_text='设备名称')
    ip = models.CharField(max_length=15, help_text='管理IP地址')
    vendor = models.CharField(max_length=16, help_text='厂商')
    device_type = models.CharField(max_length=6, help_text='设备类型')
    model = models.CharField(max_length=32, help_text='设备型号')
    sn = models.CharField(max_length=32, help_text='序列号')
    os = models.CharField(max_length=16, help_text='操作系统')
    version = models.CharField(max_length=32, help_text='版本')

    def __str__(self):
        return self.device_name

6.2 将模型同步到数据库

  • 生成迁移脚本

    (py369) [root@localhost devops]# python manage.py makemigrations hello
    Migrations for 'hello':
      hello/migrations/0004_devices.py

- 展示迁移的sql语句

(py369) [root@localhost devops]# python manage.py sqlmigrate hello 0004

BEGIN;

-- Create model Devices

此处省略...


- 执行数据库命令

(py369) [root@localhost devops]# python manage.py migrate hello
Operations to perform:

Apply all migrations: hello

Running migrations:

Applying hello.0004_devices... OK

- 查看数据库表
![](/img/bVcIpll)

- 常用命令解释

# 生产迁移脚本
python manage.py makemigrations
# 转换后的sql语句
python manage.py sqlmigrate
# 执行数据库命令
python manage.py migrate
# 所有APP及对应生效的migration
python manage.py showmigrations
# 将某个APP的migration重置
python manage.py migrate --fake hello
# 强制执行某个版本的迁移脚本
python manage.py migrate --fake hello
python manage.py migrate --fake hello 0004




# 7. ORM实现简单的增删改查

## 7.1 ORM概念

- ORM是对数据抽象建模并提供访问接口的编程方式
- 模型中的一个类(class)表示一个表(table)
- 每一个属性对应数据表中的一个字段
- 调用数据表,就是实例化类的对象

## 7.2 增 | 删 | 改 | 查

### 7.2.1 增加数据

(py369) [root@localhost devops]# python manage.py shell

In [1]: from hello.models import Devices

实例化对象

In [4]: D = Devices.objects.all()
In [5]: D

暂时还没有数据,为空

Out[5]:
In [7]: data = {'device_name':'test-sw-01', 'ip':'192.168.1.1', 'vendor':'cisco','device_type':'switch','model':'c3850','sn':'001','os':'ios','version':'15.0'}

第一种创建方式(最常用)

In [8]: D.create(**data)
Out[8]:

第二种创建方式(防止重复,速度相对较慢):

返回一个元组(对象,True或False)

In [10]: data2 = {'device_name':'test-sw-02', 'ip':'192.168.1.2', 'vendor':'cisco','device_type':'switch','model':'c3850','sn':'001','os':'ios','version':'15.0'}
In [14]: D.get_or_create(**data2)
Out[14]: (, True)
In [16]: D
Out[16]: , ]>


### 7.2.2 删除删除

数据库表中的数据(偷偷增加了一台设备):

![](/img/bVcIplm)

In [1]: from hello.models import Devices

删除一条记录

# 第一种方法:get
In [4]: D = Devices.objects.get(device_name = 'test-sw-02')
In [5]: D.delete()
Out[5]: (1, {'hello.Devices': 1})
# 第二种方法:filter
In [2]: Devices.objects.filter(device_name='test-sw-03').delete()
Out[2]: (1, {'hello.Devices': 1})

先还原数据,再删除所有的记录

In [5]: Devices.objects.all().delete()
Out[5]: (3, {'hello.Devices': 3})


### 7.2.3 修改数据

第一种方法:

In [2]: D = Devices.objects.get(device_name='test-sw-03')
In [3]: D.device_name = 'test-sw-13'
In [4]: D.save()
In [5]: Devices.objects.all()
Out[5]: , , ]>

第二种方法:

指定字段更新,偷偷去看下后台的ID是多少

In [6]: Devices.objects.filter(id=11)
Out[6]: ]>
In [7]: Devices.objects.filter(id=11).update(device_name='test-sw-03')
Out[7]: 1
In [8]: Devices.objects.get(device_name='test-sw-03')
Out[8]:

多个字段更新

In [26]: data = {'vendor':'huawei','device_type':'switch','model':'S9303','sn':'001','os':'VRP'}
In [27]: Devices.objects.filter(id=11).update(**data)
Out[27]: 1


最终效果如下(通过数据库查询):

![](/img/bVcIpln)

### 7.2.4 查看数据

- 查询多条数据 

  列表嵌套一个字典(QuerySet对象)
# 查询所有
In [30]: D = Devices.objects.all()
In [31]: D
Out[31]: , , ]>
# 每个对象及对象的属性
In [32]: D[0]
Out[32]: 
In [33]: D[0].device_name
Out[33]: 'test-sw-01'
# 切片,不支持负索引
In [34]: D[:2]
Out[34]: , ]>
# 遍历
In [36]: for d in D:
    ...:     print(d.device_name)
    ...:
test-sw-01
test-sw-02
test-sw-03
# 返回指定的字段(values_list 和 values)
In [37]: D.values_list('device_name','ip')
Out[37]: 
    
In [39]: D.values('device_name','vendor')
Out[39]: 

```
  • 查询一条数据

    # 第一种方法:
    In [2]: D = Devices.objects.get(device_name='test-sw-01')
    In [3]: D
    # 返回的是一个对象
    Out[3]: 
    # 取对象的属性值
    In [4]: D.device_name
    Out[4]: 'test-sw-01'
    In [5]: D.vendor
    Out[5]: 'cisco
    
    # 第二种方法:
    In [6]: data = {'device_name':'test-sw-01'}
    In [7]: D = Devices.objects.get(**data)
    In [8]: D.device_name
    Out[8]: 'test-sw-01'
    
    • 过滤查询
    In [9]: Devices.objects.filter(device_name='test-sw-01')
    Out[9]: ]>
    In [11]: Devices.objects.filter(**data)
    Out[11]: ]>
    
    • 过滤常用方法:
    # 不区分大小写:<属性值>__iexact
    In [16]: Devices.objects.filter(device_name__iexact='test-sw-01')
    Out[16]: ]>
    
    # 包含匹配:<属性值>__contains
    In [17]: Devices.objects.filter(device_name__contains='sw')
    Out[17]: , , ]>
    # 模糊匹配,不分区大小写:<属性值>__icontains
    In [18]: Devices.objects.filter(device_name__icontains='sw')
    Out[18]: , , ]>
    
    # 正则模糊匹配:<属性值>__regex
    In [20]: Devices.objects.filter(device_name__regex='-03$')
    Out[20]: ]>
    # 正则模糊匹配,不区分大小写:<属性值>__regex
    In [21]: Devices.objects.filter(device_name__iregex='^test')
    Out[21]: , , ]>
    
    # 排除过滤:<属性值>__contains
    In [22]: Devices.objects.exclude(device_name__contains='test-sw-01')
    Out[22]: , ]>
    # 包含带有sw的device_name,但排除了vendor是cisco厂商的
    In [23]: Devices.objects.filter(device_name__contains='sw').exclude(vendor='cisco')
    Out[23]: ]>
    
    # filter其他常用过滤查询方法
    __exact:精确匹配
    __iexact:精确匹配,忽略大小写
    __gt:大于
    __gte:大于等于
    __lt:小于
    __lte:小于等于
    __in:在一个list列表范围内
    __startswith:以...开头
    __startswith:以...开头,忽略大小写
    __endswith:以...结尾
    __range:在...范围内
    __year:日期的年份
    __month:日期的月份
    __day:日期的日数
    __isnull=True/False:字段是否为空
    

getfilter的区别:

 # 都可以获取到指定的对象;
 # get是获取唯一数据的场景,数据不存在会报错;
 # filter适用于任何场景,返回是一个QuerySet对象,数据不存在则返回是空的对象。
  • 排序查询

    # 正序
    In [32]: Devices.objects.all().order_by('device_name')
    Out[32]: , , ]>
    # 倒序,前面加
    In [33]: Devices.objects.all().order_by('-device_name')
    Out[33]: , , ]>

8. 打通MTV

8.1 创建模型

参见以上的hello/models.py的配置。

8.2 创建视图view

from django.shortcuts import render
from hello.models import Devices

def devicelist(request):
    # 对象实例化
    devices = Devices.objects.all()
    # {'devices':devices}表示传参
    return render(request, 'hello/device.html', {'devices':devices})

8.3 创建模板


{% extends "base.html" %}


{% block title %}设备列表{% endblock %}


{% block body %}

设备列表

{% for device in devices %} {% endfor %}
设备名称 IP地址 厂商 设备类型 型号 序列号 操作系统 版本号
{{ device.device_name }} {{ device.ip }} {{ device.vendor }} {{ device.device_type }} {{ device.model }} {{ device.sn }} {{ device.os }} {{ device.version }}
{% endblock%}

8.4 创建路由视图URL

from django.urls import path
from hello import view

app_name = 'hello'
urlpatterns = [
    path('devicelist', view.devicelist, name='devicelist'),
]

8.5 效果图如下:

Django之MTV实战(2)_第9张图片

大家先不要在意前端效果,后面的项目,再把UI这块优化好,先到这里了,大家学会了吗?

好不容易码完这篇了,大家点个赞吧!


如果喜欢的我的文章,欢迎关注我的公众号:点滴技术,扫码关注,不定期分享

公众号:点滴技术

你可能感兴趣的:(python)