内容目录
- 路由系统
- 模版
- Ajax
- model数据库操作,ORM
路由系统
django中的路由系统和其他语言的框架有所不同,在django中每一个请求的url都要有一条路由映射,这样才能将请求交给对一个的view中的函数去处理。其他大部分的Web框架则是对一类的url请求做一条路由映射,从而是路由系统变得简洁。
1、每个路由规则对应一个view中的函数
url(r'^index/', views.index), url(r'^login/', views.login),
当我们的页面有很多比如分页效果,有几十页甚至百页的那我们需要也要写这么多的规则吗?
django可以通过正则匹配来表匹配函数
url(r'^detail/(\d+)', views.detail), url(r'^detail2/(\d+)/(\d+)', views.detail2),
对应的views为
#因为上面的url规则中后面匹配数字,所以是相当于多一个的参数,所以在函数中我们需要带上参数来定义 def detail(request, nid): print(nid) return HttpResponse('OK') def detail2(request, xid, nnid): print(xid,nnid) return HttpResponse('OK')
正则的分组定义
#定义组名为p1和x2 url(r'^detail3/(?P<p1>\d+)/(?P<x2>\d+)/', views.detail3), #views中函数定义方式 def detail3(request, p1, x2): print(p1,x2) return HttpResponse('OK')
2、根据app对路由规则进行一次分类
#定义web开头访问的都到app01的urls文件中去匹配规则 from django.conf.urls import url, include from app01 import views#因为要调用views中的函数需要导入views文件 url(r'^web/', include('app01.urls')), #在app01目录中创建urls.py,在该urls定义规则 from app01 import views url(r'^index/(\d+)/', views.index), url(r'^detail/(\d+)/', views.detail), url(r'^template', views.template),
模板
1、模版的执行
模版的创建过程,对于模版,其实就是读取模版(其中嵌套着模版标签),然后将 Model 中获取的数据插入到模版中,最后将信息返回给用户;实际上是views中定义返回给前端页面的数据,通过前端html页面将返回来的数据展示出来。
一个简单demo来看过程
#urls文件中内容 url(r'^template', views.template), #views中内容 def template(request): return render(request, 'template.html', {'k1':'VVVV','k2':[11,22,33],'k3': {'nid':12,'name': 'jabe'}}) #html代码见下面代码
模板demo
{% load xx %} <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title></title> </head> <body> {{ k1 }} {{ k2.1 }} {{ k3.name }} {% for item in k2 %} <p>{{ item }},{{ forloop.counter }}, {{ forloop.counter0 }}, {{ forloop.first }}, {{ forloop.last }}, {{ forloop.revcounter }}</p> {% endfor %} {% if k1 == 'v1' %} <h1>V1</h1> {% elif k1 == 'v2' %} <h1>V2</h1> {% else %} <h1>7777</h1> {% endif %} {{ k1 }} {{ k1|lower }} {{ k1|f1:"jabe,123" }} {% f2 1 2 3 4 %} {% if k1|f3 %} <h1>True</h1> {% else %} <h1>False</h1> {% endif %} </body> </html>
HTML代码
2、模板语言
{{ item }} {% for item in item_list %} <a>{{ item }}</a> {% endfor %} forloop.counter forloop.first forloop.last {% if ordered_warranty %} {% else %} {% endif %} 母板:{% block title %}{% endblock %} 子板:{% extends "base.html" %} {% block title %}{% endblock %} 帮助方法: {{ item.event_start|date:"Y-m-d H:i:s"}} {{ bio|truncatewords:"30" }} {{ my_list|first|upper }} {{ name|lower }}
3、自定义simple_tag
a、在app中创建templatetags模块
b、创建任意 .py 文件,如:xx.py
#!/usr/bin/env python #coding:utf-8 from django import template from django.utils.safestring import mark_safe from django.template.base import resolve_variable, Node, TemplateSyntaxError register = template.Library() @register.simple_tag def my_simple_time(v1,v2,v3): return v1 + v2 + v3 @register.simple_tag def my_input(id,arg): result = "<input type='text' id='%s' class='%s' />" %(id,arg,) return mark_safe(result)
c、在使用自定义simple_tag的html文件中导入之前创建的 xx.py 文件名,html的最上部添加导入
{% load xx %}
d、使用simple_tag
{% my_simple_time 1 2 3%} {% my_input 'id_username' 'hide'%}
e、在settings中配置当前app,不然django无法找到自定义的simple_tag
INSTALLED_APPS = ( 'django.contrib.admin', 'django.contrib.auth', 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.messages', 'django.contrib.staticfiles', 'app01', )
通过上面的路由系统和模板我们可以做一个如下的实例
#应用下面的urls url(r'^assets', views.assets), url(r'^userinfo', views.userinfo), #views中 def assets(request): assets_list = [] for i in range(10): temp = {'hostname': 'h1'+str(i), 'port': 80} assets_list.append(temp) return render(request, 'assets.html', {'assets_list': assets_list}) def userinfo(request): user_list = [] for i in range(10): temp = {'username': 'h1'+str(i), 'salary': 80} user_list.append(temp) return render(request, 'userinfo.html', {"user_list": user_list})
用户和资产demo
{% extends 'layout.html' %} {% block body %} <table> {% for item in assets_list %} <tr> <td>{{ item.hostname }}</td> <td>{{ item.port }}</td> </tr> {% endfor %} </table> {% include 'son.html' %} {% include 'son.html' %} {% include 'son.html' %} {% include 'son.html' %} {% include 'son.html' %} {% include 'son.html' %} <h3></h3> {% endblock %} {% block js %} <script> document.getElementById('assets').className = 'active'; </script> {% endblock %}
html相关代码--assets
{% extends 'layout.html' %} {% block css %} <style></style> {% endblock %} {% block body %} <ul> {% for item in user_list %} <li>{{ item.username }},{{ item.salary }}</li> {% endfor %} </ul> {% endblock %} {% block js %} <script> document.getElementById('userinfo').className = 'active'; </script> {% endblock %}
html相关代码---userinfo
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title></title> <link rel="stylesheet" href="..." /> <style> .pg-header{ height: 48px; background-color: cadetblue; } .pg-body{ min-height: 500px; } .pg-body .body-menu{ width: 20%; float: left; } .pg-body .body-content{ width: 80%; float: left; } .pg-footer{ height: 100px; background-color: brown; } .active{ background-color: aquamarine; color: white; } </style> {% block css %}{% endblock %} </head> <body> <div class="pg-header"> 后台系统V1 </div> <div class="pg-body"> <div class="body-menu"> <ul> <li><a id="userinfo" href="/web/userinfo">用户管理</a></li> <li><a id="assets" href="/web/assets">资产管理</a></li> </ul> </div> <div class="body-content"> {% block body %}{% endblock %} </div> </div> <div class="pg-footer"></div> <script src="xxx"></script> {% block js %}{% endblock %} </body> </html>
公用模板---layout
ajax
对于WEB应用程序:用户浏览器发送请求,服务器接收并处理请求,然后返回结果,往往返回就是字符串(HTML),浏览器将字符串(HTML)渲染并显示浏览器上。
AJAX,Asynchronous JavaScript and XML (异步的JavaScript和XML),一种创建交互式网页应用的网页开发技术方案。
- 异步的JavaScript:
使用 【JavaScript语言】 以及 相关【浏览器提供类库】 的功能向服务端发送请求,当服务端处理完请求之后,【自动执行某个JavaScript的回调函数】。
PS:以上请求和响应的整个过程是【偷偷】进行的,页面上无任何感知。 - XML
XML是一种标记语言,是Ajax在和后台交互时传输数据的格式之一
利用AJAX可以做:
1、注册时,输入用户名自动检测用户是否已经存在。
2、登陆时,提示用户名密码错误
3、删除数据行时,将行ID发送到后台,后台在数据库中删除,数据库删除成功后,在页面DOM中将数据行也删除。(博客园)
ajax实例
#urls.py配置 url(r'^ajax_demo', views.ajax_demo), #views.py from django.shortcuts import render # Create your views here. from django.shortcuts import HttpResponse from django.shortcuts import render USER_LIST = [] for item in range(94): temp = {"id": item, 'username':'alex'+str(item), 'email': 'email' + str(item)} USER_LIST.append(temp) def index(request, page): print(page) # 1,0-9 # 2,10-19 # 3,20-29 page = int(page) start = (page - 1) * 10 end = page * 10 user_list = USER_LIST[start:end] # return HttpResponse('OK') return render(request, 'index.html', {'user_list': user_list}) def detail(request, nid): nid = int(nid) current_detail_dict = USER_LIST[nid] # current_detail_dict? return render(request, 'detail.html', {'current_detail_dict': current_detail_dict}) def template(request): return render(request, 'template.html', {'k1':'VVVV','k2':[11,22,33],'k3': {'nid':12,'name': 'alex'}}) def assets(request): assets_list = [] for i in range(10): temp = {'hostname': 'h1'+str(i), 'port': 80} assets_list.append(temp) return render(request, 'assets.html', {'assets_list': assets_list}) def userinfo(request): user_list = [] for i in range(10): temp = {'username': 'h1'+str(i), 'salary': 80} user_list.append(temp) return render(request, 'userinfo.html', {"user_list": user_list}) import json def ajax_demo(request): if request.method == 'POST': ret = {'status': False, 'message': ''} user = request.POST.get('user',None) pwd = request.POST.get('pwd',None) if user == '111' and pwd == '222': ret['status'] = True return HttpResponse(json.dumps(ret)) else: ret['message'] = "用户名或密码错误" return HttpResponse(json.dumps(ret)) return render(request, 'ajax_demo.html')
ajax--demo
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title></title> </head> <body> <div> <p>用户名:<input type="text" id="username" /></p> </div> <div> <p>用户名:<input type="password" id="pwd" /></p> </div> <input type="button" value="提交" onclick="SubmitForm();" /> <script src="/static/jquery-1.8.2.min.js"></script> <script> function SubmitForm(){ $.ajax({ url: '/web/ajax_demo/', type: 'POST', data: {'user': $('#username').val(), 'pwd': $('#pwd').val()}, dataType: 'json', success: function (data) { // data = 字符串 {status:xx,message:''} // data对象 //var data_dict = JSON.parse(data); if(data.status){ location.href = "http://www.baidu.com"; }else{ alert(data.message); } } }) } </script> </body> </html>
ajax_demo.html
其他
1、XMLHttpRequest对象(原生Ajax)
2、jQuery - XMLHttpRequest对象(jQuery ajax)
3、跨域Ajax
浏览器同源策略:
Ajax,只能给自己的域名发请求
跨域名,浏览器做拦截
model,ORM
到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞:
- 创建数据库,设计表结构和字段
- 使用 MySQLdb 来连接数据库,并编写数据访问层代码
- 业务逻辑层去调用数据访问层执行数据库操作
创建数据库生成库表命令
python manage.py makemigrations
python manage.py migrate
Django创建admin用户命令
python manage.py createsuperuser
django中遵循 Code Frist 的原则,即:根据代码中定义的类来自动生成数据库表。
1、创建Model,之后可以根据Model来创建数据库表
from django.db import models class userinfo(models.Model): nid = models.AutoField(primary_key=True) name = models.CharField(max_length=30, verbose_name='用户名',editable=False) email = models.EmailField(db_index=True) memo = models.TextField() img = models.ImageField(upload_to='upload') user_type = models.ForeignKey("UserType", null=True, blank=True)# unique # user_type = models.OneToOneField("UserType", null=True, blank=True)# unique # ctime = models.DateTimeField(auto_now_add=True) # uptime = models.DateTimeField(auto_now=True) # gender = models.ForeignKey(Gender) gender_choices = ( (0, "男"), (1, "女"), ) gender = models.IntegerField(choices=gender_choices,default=1)
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 文件
字段说明
1、null=True 数据库中字段是否可以为空 2、blank=True django的 Admin 中添加数据时是否可允许空值 3、primary_key = False 主键,对AutoField设置主键后,就会代替原来的自增 id 列 4、auto_now 和 auto_now_add auto_now 自动创建---无论添加或修改,都是当前操作的时间 auto_now_add 自动创建---永远是创建时的时间 5、choices GENDER_CHOICE = ( (u'M', u'Male'), (u'F', u'Female'), ) gender = models.CharField(max_length=2,choices = GENDER_CHOICE) 6、max_length 7、default 默认值 8、verbose_name Admin中字段的显示名称 9、name|db_column 数据库中的字段名称 10、unique=True 不允许重复 11、db_index = True 数据库索引 12、editable=True 在Admin里是否可编辑 13、error_messages=None 错误提示 14、auto_created=False 自动创建 15、help_text 在Admin中提示帮助信息 16、validators=[] 17、upload-to
更多参数
2、连表关系:
- 一对多,models.ForeignKey(ColorDic)
- 一对一,models.OneToOneField(OneModel)
- 多对多,authors = models.ManyToManyField(Author)
应用场景: 一对一:在某表中创建一行数据时,有一个单选的下拉框(下拉框中的内容被用过一次就消失了)。 例如:原有含10列数据的一张表保存相关信息,经过一段时间之后,10列无法满足需求,需要为原来的表再添加5列数据。 一对多:当一张表中创建一行数据时,有一个单选的下拉框(可以被重复选择)。 例如:创建用户信息时候,需要选择一个用户类型【普通用户】【金牌用户】【铂金用户】等。 多对多:在某表中创建一行数据是,有一个可以多选的下拉框。 例如:创建用户信息,需要为用户指定多个爱好。
3、数据库操作
- 增加:创建实例,并调用save
- 更新:a.获取实例,再sava;b.update(指定列)
- 删除:a. filter().delete(); b.all().delete()
- 获取:a. 单个=get(id=1) ;b. 所有 = all()
- 过滤:filter(name='xxx');filter(name__contains='');(id__in = [1,2,3]) ;
icontains(大小写无关的LIKE),startswith和endswith, 还有range(SQLBETWEEN查询)'gt', 'in', 'isnull', 'endswith', 'contains', 'lt', 'startswith', 'iendswith', 'icontains','range', 'istartswith' - 排序:order_by("name") =asc ;order_by("-name")=desc
- 返回第n-m条:第n条[0];前两条[0:2]
- 指定映射:values
- 数量:count()
- 聚合:from django.db.models import Min,Max,Sum objects.all().aggregate(Max('guest_id'))
- 原始SQL
cursor = connection.cursor() cursor.execute('''SELECT DISTINCT first_name ROM people_person WHERE last_name = %s""", ['Lennon']) row = cursor.fetchone()
参考url:
http://www.cnblogs.com/wupeiqi/articles/5237704.html
http://www.cnblogs.com/wupeiqi/articles/5246483.html
http://www.cnblogs.com/wupeiqi/articles/5703697.html