Django教程
Python一直是我最喜欢的语言,在这个寒假打算认真学习一下Python的Web框架。在Django和Tornado之间我选择了前者,没有特别的原因,网上人云亦云的,肯定不会有一方离另一方差很远,我就直接去看了看Github上两个项目的活跃度,所以选择了前者。
应该说Django坚持自己造轮子,确实为开发者节约了不少的时间,我很看重它的扩展功能,packages数量十分丰富。Django采用的是最流行也是我最熟悉的MVC设计模式,虽然在之前的一个PHP(Laravel)项目中也是采用的MVC模式,但一直都没怎么吃透,始终在各层分离的时候不是很清晰,所以也可趁学习Django对MVC的概念进行强化。
Django另一个我特别喜欢的特性就是Application,它与Project的概念不同,一个APP就相当于一个功能模块,一个Project可以包含多个APP,一个APP可以同时被多个Project引用,App增加了代码的复用机会,提高了扩展性和松耦合性,Django中很多的packages都是以APP的形式存在的。
另外,我学习主要参考的是开源书籍 Django搭建简易博客教程
下面是搭建一个Django环境的基本步骤:
新建项目
django-admin startproject 项目名
这样会在当前目录新建一个目录,里面已经有一些基本的配置文件:django_test ├── db.sqlite3 ├── django_test │ ├── __init__.py │ ├── settings.py │ ├── urls.py │ └── wsgi.py └── manage.py
新建APP
django-admin startapp APP名称
如果是搭建一个非常简单的应用,那么不使用APP也行,只需要吧路由指向目标view就可以了,但是如果要搭建复杂的应用并且需要良好的隔离性,那最好使用APP。同样,使用该命令也会在当前目录下新建一个目录,里面已经包含一些配置文件:django_test/testapp ├── admin.py # 注册models,用于admin管理 ├── __init__.py ├── migrations # 数据库迁移 ├── models.py # 定义models ├── tests.py # 单元测试 ├── apps.py # App的配置类,AppConfig用于存储应用程序的元数据 └── views.py # 视图文件
如过添加了APP,那么需要在主配置文件
settings.py
里面的INSTALLED_APPS
里面添加该APP的名称Hello World! 所有入门教程都必须要有一个Hello World! 首先,在APP的视图文件views.py里添加函数,该函数直接返回一个字符串的响应:
from django.shortcuts import renderfrom django.http import HttpResponsedef hello(request): return HttpResponse('Hello World!')
然后添加URL,在Project目录里的
urls.py
里进行管理,添加hello的url如下:from django.conf.urls import patterns, include, url
from django.contrib import adminurlpatterns = [
url(r'^admin/', include(admin.site.urls)),
url(r'^hello/', 'testapp.views.hello'),
]运行
python manage.py runserver
如果要以daemon的方式在后台运行,可以使用nohup命令nohup python manage.py runserver 0.0.0.0:8000 &
使用它可以打开Django自带的默认Web引擎,可以在 http://127.0.0.1:8000中查看 在测试的时候可以使用该引擎,它不仅轻量,而且在打开后还会自动检测代码的更改,进行自动更新,这样就不用每次对代码变动了都来重启一次
配置项
全局配置
需要注意的是,Django官方并没有默认的分离配置文件的方案,我觉得最佳的方式是,建立多个配置文件,然后在默认的配置文件里面进行导入即可。例如:
env = 'local'if env == 'local': from settings_local import *else: from settings_prod import *
配置文件内容
DEBUG = True # DEBUG模式TEMPLATE_DEBUG = True # TEMPLATE的DEBUG模式ALLOWED_HOSTS = [] # 设置哪些域名可以访问,当debug为false时必须为其指定一个值,['*']表示允许所有的访问INSTALLED_APPS = [默认APP+自己的APP] MIDDLEWARE_CLASSES = [中间件] ROOT_URLCONF = 'admin.urls' # 读取的默认的url文件# Database 数据库的配置DATABASES = { 'default': { 'ENGINE': 'django.db.backends.mysql', # MySQL/Mariadb数据库设置 'NAME': 'admin', 'USER': 'root', 'PASSWORD': 'mysql', 'HOST': '127.0.0.1', 'PORT': 3307 } } LANGUAGE_CODE = 'en-us' # 语言,中文可用zh-Hans、zh-CN,完整列表见:http://www.i18nguy.com/unicode/language-identifiers.htmlTIME_ZONE = 'Asia/Chongqing' # 时区USE_TZ = False # 数据库中要使用时间戳就应该关闭这个# Static files (CSS, JavaScript, Images) 静态文件目录 STATIC_URL = '/static/'STATIC_ROOT = os.path.join(BASE_DIR, 'static') # 这个选项默认是没有的,在编码时将静态文件放在APP中的static目录下,部署时用python manage.py collectstatic就可以把静态文件收集到STATIDC_ROOT目录
在其他文件访问全局配置项,可以这样访问:
from django.conf import settings settings.DEBUG
应用配置
在上面新建的app的目录结构里面又一个apps.py
文件,它存储了应用的元数据,通过继承AppConfig
来配置其属性,可配置的选项如下:
可配置的属性 AppConfig.name # 应用的完整Python路径,例如django.crontrib.admin,在整个Django项目中必须是唯一的 AppConfig.label # 应用的缩写,例如admin, AppConfig.verbose_name # 应用的适合阅读的名称 AppConfig.path # 应用目录的文件系统路径,例如/usr/lib/python3.4/dist-packages/django/contrib/admin 可配置的方法 AppConfig.get_models() # 返回可迭代的Model类 AppConfig.get_model(model_name) # 返回具体的Model AppConfig.ready() # 执行初始化任务
请求与响应
HttpResponse('字符串', content_type="text/plain") # 指定content_type的响应 HttpRequest.method # 请求种类HttpRequest.GET # 获取所有的GET参数(字典)HttpRequest.POST # 获取POST的参数(字典)HttpRequest.scheme # 表示请求的模式,是http还是httpsHttpRequest.cookies # 包含了所有的cookie信息HttpRequest.session # session信息HttpRequest.FILES # 包含了上传的文件HttpRequest.meta # 包含了http请求的各种headersHttpRequest.user # 当前的登录的用户,配合着auth使用get_host() #不解释了吧get_full_path() # 获取路径,不包含域名build_absolute_uri() # 获取完整路径is_secure() # 如果是https返回true,否则falseis_ajax() # 是否是ajax请求return JsonResponse(error, status = 422) # 返回指定状态码ip = request.META.get('REMOTE_ADDR') # 获取用户IPreturn HttpResponseRedirect('/') # 重定向
路由与视图
url: web访问请求的入口(相当于Laravel里的路由)
view:应用的逻辑部分,从客户端接收请求,处理并返回数据,一般返回到template模板进行渲染(相当于Laravel里的控制器)
将/test
定位到article这个APP里面的views里面的home方法来处理的形式
url(r'^test$', 'article.views.home')
url传递参数
Django的路由是采用正则表达式来匹配的,同样能使用命名组,比如(?P<name>)
,这样就可以通过URL给views传递参数了,例如:
# 有如下视图def hello(request, name): return HttpResponse('name is %s', % name)# 在url中可以这样写url(r'^(?P<name>\d+)/$', 'testapp.views.hello)
url命名
url(r'^add/$', 'app.views.add', name='add')
给url命名可以方便我们进行统一修改url样式,比如之前用a-and-b
的url可以访问到add这个方法,但现在如果想改成a/b
的方式来访问,那么由于后端的模板渲染等都是用的其命名add
,就无需修改后端逻辑了
路由按照app分组
首先主urls.py里面使用include包含应用下的urls.py文件
from django.conf.urls import include urlpatterns = [ url(r'^app1', include('app1.urls')), ]
然后在app1下的urls.py文件里面,进行如下设置:
from django.conf.urls import urlfrom . import views urlpatterns = [ url(r'^/$', views.hello), ]
这样就可以通过/app1/
,来访问app1下的hello方法了
数据库
Django同很多框架一样使用了ORM(Object Relational Mapping,对象关系映射)的方式,每个model类代表一张数据库的表,每一个属性代表其一个字段(这种特性的实现依赖于python的元类)。
数据表定义
定义model的文件是project/app/models.py
里面,例如,要定义一张用户表:
fromo django.db import modelsclass User(models.Model): username = models.CharField(max_length = 20) # 用户名字段 create_time = models.DateTimeField(auto_now_add = True) # 注册日期字段,如果同时有两个字段对应着同一个外键,那么久得重命名字段名了,比如: receiver = models.ForeignKey(Users, null=True, related_name='receiver') poster = models.ForeignKey(Users, null=True, related_name='poster') def __str__(self): '''这个函数可以用于str(obj)函数来输出该对象的信息,默认是表名''' return self.username class Meta: db_table = '自定义表名' unique_together = ('column_1', 'column_2') # 联合唯一键
当建立好models过后,执行如下命令就可以在数据库中新建或更新数据表了:
python manage.py makemigrationspython manage.py migrate
注:如果是有修改的,那么新添加的数据必须要允许null或者设置默认值,否则会报错,这其实是为了保护已经存在了的数据,当然在添加完该字段后把null去掉再更新数据库就可以了。
字段类型
注:Django默认为每张表设置了一个int(11)的自增主键,不需要自己去定义了。
字段通用参数
primary_key = False/True # 是否设置为主键blank = False/True # 是否可为空,这其实是用于Field的判断null = False/True # 是否可为空,这才是真正的数据库里面是否可以为nullmax_length = 3 # 最大长度,对整型无效default = '' # 设置默认值verbose_name = # 相当于备注,如果没给出那么就是该字段,当然,要指定的话,可以直接第一个参数一个字符串就可以指定了editable = False/True # 是否可编辑unique = False/True # 是否唯一auto_now = False/True # 用于时间,每次更新记录的时候更新该字段auto_now_add = False/True # 用于时间,创建新纪录的时候自动更新该字段choices # 很实用的一个功能,相当于存储一个枚举列表,其中左边的key是实际存储在数据库中的值,例如,可以这样定义一个字段: YEAR_IN_SCHOOL_CHOICES= ( ('FR', 'Freshman'), ('SO', 'Sophomore'), ('JR', 'Junior'), ('SR', 'Senior'), ) 然后在定义字段的时候给个参数choices=YEAR_IN_SCHOOL_CHOICES,在插入字段的时候,使用'RF'这样的,在获取字段值的时候这样:p.get_year_in_school_display()即可显示'Freshman'verbose_name # 定义字段的注释
常用类型
# 数字类型:AutoField # 自增长字段IntegerField # 长度为11的整数PositiveIntegerField: SmallIntegerField PositiveSmallIntegerField BigIntegerField: BinaryField: BooleanField: NullBooleanField: DecimalField(max_digits = None, decimal_places = None) FloatField# 字符类型CharField # 字符串类型,可用max_length指定长度,枚举类型也使用该方式,只需要指定枚举枚举元组即可,例如type = models.CharField('类型', choices=CONTENT_TYPE),其中CONTENT_TYPE=(('a', 'abc'))TextField:text类型 CommaSeparatedIntegerField:用逗号分隔的整数,我擦,这有用# 时间类型DateField # DATE类型TimeField # datetime.time,时间DateTimeField() # DATETIME类型,包括了日期和时间,需要注意的是Django默认的TIME_ZONE是UTC,在初始化的时候,格式如"2015-04-27T15:01:00Z",它属于python里面的datetime.datetime类型,可分别用year/month/day等获取时间。另外Django如果要实用MySQL里面的TIMESTAMP类型也是用该字段表示,并且在插入的时候不能直接插入一个整数,依然只能插入一个datetime.datetime对象,用时间戳的时候USE_TZ必须为Falseunique_for_date属性:比如,如果有一个title字段,其有一个参数unique_for_date = "pub_date",那么该表就不会出现title和pub_date同时相同的情况,其它的还有unique_for_month,unique_for_year 其它很有用的类型: EmailField:Email邮箱类型 FileField:文件类型,不过不能设置为primary_key和unique,要使用该字段还有很多需要注意的地方,具体的见官方文档 FilePathField:同上 ImageField IPAddressField:从1.7开始已经不建议使用了,应该使用下面这个 GenericIPAddressField: URLField UUIDField
数据库SQL操作
要使用model,必须先导入它,例如from app.models import Blog
,一条记录就是一个model类的实例,要获取其字段值,直接用点号访问即可,例如有Blog对象blog,那么可以直接用blog.userName访问其值。
如果想要在执行数据库操作的时候查看数据库的实际的SQL语句,可以这样
from django.db import connection print(Blogobjects.filter(name="").query) # 这样可以将SQL语句打印出来connection.queries # 会返回一个所有执行过的SQL的列表,并且每条时一个字典,包含了SQL语句以及SQL所执行的时间
查询记录
Blog.objects.all() # 获取该表的所有记录,返回的是记录对象组成的列表Blog.objects.get(pk=1) # 根据主键获取数据Blog.objects.get(name="") # 只会找到第一个匹配的数据Blog.objects.filter(name="")# 这个就会找到匹配的多个数据Blog.objects.filter(name__contains="") # 模糊查找name字段的值,返回列表Blog.objects.order_by("字段1", "字段2") # 排序,order_by不加任何参数表示不需要排序Blog.objects.all().order_by("字段") Blog.objects.count() # 返回记录总数Blog.objects.values('id', 'name') # 相当于select id name from Blog,返回的事是一个字典Blog.objects.values('name').distinct() # distinct在django的mysql引擎中无法对field进行distinct操作,所以需要这样做Blog.objects.values_list('id', flat=True)# 查询该字段的所有值并且返回的是id的列表,而不是包括了名字的字典Blog.objects.all().defer('title') # 仅仅取某个字段,这里返回是一个model对象Blog.objects.all().only('title') # 仅仅取某个字段,也是返回一个model对戏那个Blog.objects.all().values_list('title') # 仅仅取某个字段,这里返回一个数组Blog.objects.latest('id') # 根据某个字段查找其最后一条记录,返回的是一个对戏那个,不是idBlog.objects.filter(time__gte = '2015-07-23', time__lte = '2015-07-24') # 大于等于并且小于等于,不加e表示不能等于Blog.objects.filter(time__isnull = True)# 判断某个字段是否为空Blog.objects.all().exclude(id=7) # 排除,即不等于Blog.objects.filter('time__year': '2015', 'time__month': '08', 'time__day': '17'):按年月日查询日期,可仅查询其中某一个条件# Q查询,可以对关键字参数进行封装,可以使用&,|,~等操作from django.db.models import Q Blog.objects.filter( Q(name__startswith='wang') | ~Q(name__startswith='hao') ) Blog.objects.get( Q(name__startswith='wang'), Q(name__startswith='hao')) # 逗号就可以直接表示and了 print(People.objects.filter( (Q(birth_lunar__month=old_month) & Q(birth_lunar__day=old_day)) | (Q(birth_new__month=new_month) & Q(birth_new__day=new_day)) ).query)
新增记录
post = Blog(userName="wanghao", userId=12) post.save()# 批量插入/新增posts = []for i in title: posts.append(Posts(title = title)) Posts.objects.bulk_create(posts)
更新记录
post = Blog.objects.filter(id=1).update(userName="new") #1.7之前更新单条记录如果要用字典的话就只能这样了Blog.objects.all().update(userName="new") # 还可以批量更新obj, created = Posts.objects.update_or_create(pk = 3, title='wang', defaults = updated_values) # 1.7之后可以用这种方法来更新或者创建一个,如果没找到对象,那么就新建,新建或者更新的字典是defaults的值,返回值中,obj表示该对象,created是一个布尔值get_or_create(title='wang', defaults=\{\}):获取或者新建
删除记录
Blog.objects.get(userName="xiao").delete() Blog.objects.all().delete() Blog.author.through.objects.filter(author = author.id).delete() # 删除多对多关系,仅仅是删除关系,而不是删除对象
数据约束
ForeignKey
例如:
# modles.pyclass System(models.Model): name = models.CharField(max_length = 20)class Server(models.Model): ip = models.GenericIPAddressField(default = '127.0.0.1') system = models.ForeignKey(System)# views.py里面这样子使用server = Server.objects.get(id=1) server_system = server.system.name # 这样就可以获取到那个name了
OneToMany(hasMany)
一对多关系,同样使用ForeighKey实现,例如
# 在models.py中定义class Posts(models.Model): title = models.CharField('标题', max_length = 50)class Comments(models.Model): post = models.ForeignKey(Posts, related_name = 'comments_set')# 在views.py中这么用post = Posts.objects.get(pk = 1) # 获取一篇文章comments = post.comments_set.all() # 获取该文章的所有评论,是一个列表
ManyToManyField
多对多关系,有一种特殊情况,如果需要对这种关系添加额外的字段,可以使用through,添加额外的表来表示,例如,用户一张表,被使用的物品一张表,用户与物品是多对多的关系,但是有时候我们需要记录下用户使用该物品的一些其他属性,比如使用了多少次什么的,这时候就需要给这个多对多关系添加额外的字段来表示,那就需要添加额外的表了,示例如下:
class User(models.Model): username = models.CharField(max_length = 20) goods = models.ManyToManyField('物品', 'Goods', through='user_goods')class Goods(models.Model): goodsname = models.CharField(max_length = 20)class user_goods(models.Model): user = models.ForeignKey(User) goods = models.ForeignKey(Goods) clicks = models.IntegerField('点击量', default=0)
OneToOneField
必须是一对一,而不是多对一或一对多
分页
Django使用内建的paginator模块进行分页的操作,十分方便。使用方法见例子:
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger # 导入模块def listing(request): contact_list = Contacts.objects.all() # 获取所有model对象 paginator = Paginator(contact_list, 25) # 第二个参数是每页显示的数量 page = request.GET.get('page') # 获取URL参数中的page number try: contacts = paginator.page(page) except PageNotAnInteger: # 若不是整数则跳到第一页 contacts = paginator.page(1) except EmptyPage: # 若超过了则最后一页 contacts = paginator.page(paginator.num_pages) return render_to_response('list.html', {"contacts": contacts})
虽然contacts是一个Page对象,但是在模板中仍然可以使用for循环对其进行遍历,它其实是一个对象所组成的list。下面是分页按钮html模板例子:
<nav> <ul class="pagination"> <li class="{% if current_page == 1 %}disabled{% endif %}"><a href="#" aria-label="Previous"><span aria-hidden="true"></span></a></li> {% for index in page_index %} {% if index == current_page %}class="active"{% endif %}<a href="#";{{ index }}<span class="sr-only"(current)/</span></a></li> {% endfor %} <li class="{% if current_page == num_pages %}disabled{% endif %}<a href="#" aria-label="Previous"><span aria-hidden="true"></span></a></li> </ul> </nav>
初始化数据
为了方便迁移,让别人使用你的APP,有时候需要为APP里面的表提供demo数据,这时候就需要预先填充一些数据.这里使用Django的fixtures方式填充(Django提供两种填充方式)。使用JSON格式,我们可以首先使用manage.py dumpdata data.json
方式到处原来数据库中内容看看该格式,类似如下:
[ { "fields": { "userName": "小豪", "title": "第一篇文章", "userId": 1, "update": "2015-04-27T15:01:03Z", "datetime": "2015-04-27T15:01:00Z", "content": "这是文章的内容" }, "model": "digital.blog", "pk": 1 }, { "fields": { "userName": "笑总", "title": "第二篇文章", "userId": 2, "update": "2015-04-28T15:01:03Z", "datetime": "2015-04-28T15:01:03Z", "content": "这是文章的内容吗" }, "model": "digital.blog", "pk": 2 }, ]
我们可以自己按照这个模板新建填充数据,其中pk指的是主键值。当建立好json文件过后,执行python manage.py loaddata data.json
即可导入数据。
Template: Django模板
和所有的MVC框架一样,模板功能是必须有的。这里介绍一下Django模板的使用方法。
模板定义
为了方便管理,最好在app的目录下新建templates文件夹用于存放模板文件,然后在project的配置文件settings.py中指明模板文件夹的位置:
TEMPLATES['DIRS']这个变量中添加即可,比如'DIRS': [ os.path.join(BASE_DIR, 'dashboard/templates').replace('\\', '/'), ]
这样,在该app的view中就可以这样使用templates下的test.html模板文件了。例如:
def test(request): return render(request, 'test.html')
参数传递
要向模板中传递参数,可以给render添加第三个参数,该参数其实是一个字典,在模板中可以直接使用该字典的key,例如:
return render(request, 'test.html', {'name1': value1, 'name2': value2} )
这样,在模板文件test.html中就可以直接{{ name1 }}
来使用name1
的值了。
继承与引用
模板方便之处就是可以使用继承将代码分块并且将重复的地方都写在一个base.html里。当要实现继承的时候在html文件第一行写上{% extends 'base.html' %}
,然后分别实现其区块即可。 在base模板中一般这样定义区块:
{% block 块名 %} 这里直接写html代码 {% endblock %}
如果子模块没有定义某个block的内容,那么就采用父模板的,如果需要使用父模板的内容可以用{{ block.super }}
模板也可以通过引用其它模板的代码,例如,在要引用的地方使用:
{% include 'nav.html' %} {% include 'includes/nav.html' %}
静态文件css、js、img
静态文件一般当然是要存放在自己的app里面,这时候,就应该指定静态文件的路径,在project的配置文件settings.py中添加如下配置:
STATICFILES_DIRS = (
os.path.join(BASE_DIR, 'static'),
'f:/project/app/static', # 无论是windows还是linux都要用左斜杠哟
)
在模板中使用静态文件就这样:
{% load staticfiles %}
<link href="{% static 'css/style.css' %}" rel="stylesheet">
模板标签
Django内置了一些比较常用又实用的标签:
# 变量## 通过下标获取列表变量的值{{ names.0 }}# 注释{# 单行注释 #}{% comment %}多行注释{% endcomment %}# url路由有这样一个路由 url(r'^blog/', 'myapp.views.blog'), # 博客页面那么就这样使用 {% url 'digital.views.blog' %} 或者使用别名,例如 url(r'^blog/', 'myapp.views.blog', name='blog'), {% url 'blog' %} 如果是其他app的url,那么需要带上该app的namespace,首先在定义的时候需要添加namespace,如 url(r'^oauth/', include('oauth.urls', namespace='oauth')) 然后在实用url的时候: {% url 'oauth:hello' %}# for循环{% for <element> in <list> %}{% endfor %} {% for <element> in <list> reversed%}{% endfor %} 反向迭代列表 {% for <element> in <list> %}{% empty %}{% endfor %} 列表为空时的输出内容 {{ forloop.counter }} # 获取当前索引,默认从1开始{{ forloop.counter0 }} # 获取当前索引,从0开始# if语句{% if <element> %} {% elif <element> %} {% else %} {% endif %} {% ifequal 变量1 变量2 %} 比较值 {% endifequal %} ifnotequal同上
过滤器
可以直接格式化输出,是一种最便捷的转换变量输出格式的方式。
{{ today | data: "F j, Y" }}
这里可以将today这个变量直接按照规定格式输出。
这里是常见的过滤器:
add:将该数字加上一个数字,例如 `{{ value|add:"2" }}`,如果原来的值为4,那么新的值就为6,不仅进可以作用与int,还能作用与列表,将列表中每个值都加 addslashes:添加反斜杠到需要转义的地方前 capfirst:第一个字母大写 center:在字符串前后加空格,并让该字符串位于中间,例如 `{{ value|center: "5" }}`,那么输出时前后都是5个空格 cut:去除字符串中的指定字符,例如`{{ value|cut:" " }}`,表示去除字符串中的所有空格 date:按指定的格式字符串参数格式化date或者datetime对象,参数太多了,看[文档](https://docs.djangoproject.com/en/1.8/ref/templates/builtins/)default:参数没赋值的情况下给定默认值,例如`{{ value|default: "nothing" }}` default_if_none:如果它是none就给定默认值 dictsort:将一个字典组成的列表排序,例如`{{ value|dictsort:"name" }}`意思是将value这个列表里面的字典按照字典里的name的顺序来排序 dictsortreversed:与上面顺序相反 divisibleby:是否能被整除,返回True/False,例如`{{ value|divisibleby:"3" }}` filesizeformat:-h方式输出文件的大小,例如`{{ value|filesizeformat}}`,如果value=123456789,那么输出将是117.7MB first:返回列表的第一个值 floatformat:设置浮点数的显示形式 get_digit:获取一个整数的倒数第几个数字,例如`{{ value|get_digit:"2" }}`,那么123456789的值为8join:将一个列表的值添加一个分隔符并以字符串形式输出,例如`{{ value|join:"//"}}`那么['a', 'b', 'c']输出将是"a//b//c"last:返回列表的最后yield值 length:返回变量的长度,也可以在if语句里面使用,例如 {% if messages|length >= 100%} ...{% endif %} length_is:判断一长度是否是某个值,例如`{{ vlaue|length_is:"4" }}`如果value长度是4那么就返回Truelinebreaks:替换换行符,例如如果value的值是Joel\nis a slug,那么输出就是<<p>Joel<br /> is a slug</p> linenumbers:在输出的tex前加上标号 ljust:在字符串后面加上指定长度的宽度,例如`{{ value|ljust:"10" }}` make_list:将整型或字符串转换为单个单个的列表元素组成一个列表 random:在列表里面随机选取一个元素 lower:转换为小写 rjust:在字符串前面加上指定长度的宽度 slice:列表分片,例如`{{ some_list|slice:":2"}}`就表示前面两个元素妈 slugify: stringformat: striptags:取出HTML中的tag,只去内容 time:同date timesince: timeuntil title:将一个字符串转换为title的形式,即一般的第一个字母大写那种标题 truncatewords: "30":表示只显示前面30个字符 truncatechars_html: truncatewords:显示前面多少个字符,单位是词,而不是字符 truncatewords_html: unordered_list: upper:转换为大写 urlencode:将url进行编码,例如"http://www.example.org/foo?a=b&c=d"被编码为“http\%3A//www.example.org/foo\%3Fa\%3Db\%26c\%3Dd”urlize: urlizetrunc: wordcount:统计字符串中单词的数量 wordwrap:指定特定的长度来分隔字符串 yesno:
其他标签
# autoescape标签{% autoescape on %} # 去掉自动转义{{ body }} {% endautoescape %}# cycle标签:每次使用该标签,标签中的值就会变化,比如下面这个,第一次该值为row1,第二次则为row2,第三次又变为了1,感觉可以用于循环里面的奇偶什么的{% for o in some_list %} <tr class="{% cycle 'row1' 'row'2 %} ... </tr> {% endfor %} # now 标签,直接将当前时间按指定格式输出: {% now "jS F Y H:i" %} # spaceless标签:移出HTML tags之间的空白 {% spaceless %} <p> ... </p> {% endspaceless %} # verbatim:停止模版引擎,一般用于在模板里面写Javascript什么的 {% verbatim %} ... {% endverbatim %} # with标签:和语法里面的with类似 {% with total=business.employees.count %} {{ total }} employee{{ total|pluralize }} {% endwith %}
用户认证系统
Django项目默认添加了用户认证系统的,可以通过
python manage.py makemigrationspython manage.py migrate
将认证系统的数据表添加到数据库中去.
创建用户
from django.contrib.auth.models import User user = User.objects.create_user('john', 'lennon@thebeatles.com', 'johnpassword') user.save()
创建超级用户
$ python manage.py createsuperuser --email=896499825@qq.com --username=wanghaoPassword:Password (again): Superuser created successfully.
验证用户登录
需要注意的是,为了不与login冲突,views最好不要写成login
from django.contrib.auth import authenticate, login user = authenticate(username='hao', password='test')if user is not None: if user.is_activate: print('用户验证并登录成功') login(request, user) # 这才是登录,才会写入session else: print('密码正确,但是用户无法登录') return HttpResponse('居然有这个用户')else: return HttpResponse('用户不存在')
若登录成功,则会返回一个user对象,否则返回None,但是Django默认的认证系统只能认证username和password,却不能认证其它的字段,比如
email字段,看似非常糟糕,但是Django却提供了十分方便的功能来扩展默认的验证组件。查看文档:Customizing authentication
in Django
如果要验证用户名或email可以这样做:首先,在myapp目录下新建一个文件,姑且取名叫backends.py
,其内容如下:
from django.conf import settings from django.contrib.auth.models import User class EmailOrUsernameModelBackend(object): def authenticate(self, username=None, password=None): if '@' in username: kwargs = {'email': username} else: kwargs = {'username': username} try: user = User.objects.get(**kwargs) if user.check_password(password): return user except User.DoesNotExist: return None def get_user(self, user_id): try: return User.objects.get(pk=user_id) except User.DoesNotExist: return None</pre>
然后在settings.py中添加如下代码:
# 该字段指定了默认的验证后台,从上到下顺序验证,如果上面验证不成功就验证下面的AUTHENTICATION_BACKENDS = ( 'myapp.backends.EmailOrUsernameModelBackend', # 自定义的认证后台 'django.contrib.auth.backends.ModelBackend', # 这是默认的认证后台)
这样,就可以依然使用刚才的代码对用户登录进行验证了。
退出登录
from django.contrib.auth import logoutdef logout_view(request): logout(request)
限制登录用户访问路由
某些路由只能登录用户才能访问,那么只需要添加这个装饰器:
from django.contrib.auth.decorators import login_required@login_requireddef my_view(request): ......
未登录的用户将会重定向到settings.LOGIN_URL
去
Channels
用于与websockets通信
signal
参考django中signal与操作系统的signal是完全不一样的.Django的signal是一种同步的消息队列.通常在以下情况进行使用:
signal的receiver需要同时修改对多个model时
将多个app的相同signal引到同一receiver中处理时
在某一model保存之后将cache清除时
无法使用其他方法, 但需要一个被调函数来处理某些问题时
作为网站的通知
django-crontab插件
Django下的定时任务插件,依赖于Linux的cron服务。
安装
pip install django-crontab
进行安装,然后在django的配置文件中添加一个APP
INSTALLED_APPS = ( 'django_crontab', ... )
使用
编写完定时任务逻辑以后,需要在配置文件中添加上,例如
CRONJOBS = [ ('*/5 * * * *', 'myapp.cron.my_scheduled_job') ('*/5 * * * *', 'myapp.cron.other_scheduled_job', ['arg1', 'arg2'], {'verbose': 0}), ('0 4 * * *', 'django.core.management.call_command', ['clearsessions']), ]
最后,将其添加到系统cron
服务中去
python manage.py crontab add # 将当前配置文件中的定时任务添加到cron中去,当add以后,会在crontab -e里面出现类似这样的一条记录# */5 * * * * /usr/local/bin/python /usr/src/app/manage.py crontab run e0418752956c4dd997212171486888ff # django-cronjobs for admin# 前面是命令部分,后面是根据定义的函数计算出来的hash值,最后面则是自带的注释python manage.py crontab show # 列出当前已经添加到cron中的定时任务 python manage.py crontab remove # 移除所有的定时任务
作者:豪翔天下
链接:https://www.jianshu.com/p/bc6d8f03eacf