Django基本使用
1 安装
1.1 安装pip
wget https://bootstrap.pypa.io/get-pip.py
python get-pip.py
1.2 安装django
pip install django==1.10.6
2 创建项目
2.1 使用 管理工具 django-admin.py 来创建 PyLearn 项目:
django-admin.py startproject PyLearn
# root @ VM_33_19_centos in /data/PyLearn [10:21:21]
$ tree
.
|-- manage.py
`-- PyLearn
|-- __init__.py
|-- settings.py
|-- urls.py
`-- wsgi.py
1 directory, 5 files
- manage.py: 一个实用的命令行工具,可让你以各种方式与该 Django 项目进行交互。
- PyLearn/init.py: 一个空文件,告诉 Python 该目录是一个 Python 包。
- PyLearn/settings.py: 该 Django 项目的设置/配置。
- PyLearn/urls.py: 该 Django 项目的 URL 声明; 一份由 Django 驱动的网站"目录"。
- PyLearn/wsgi.py: 一个 WSGI 兼容的 Web 服务器的入口,以便运行你的项目。
2.2 启动服务
python manage.py runserver 0.0.0.0:8000
问题:
$ /usr/local/bin/python manage.py runserver 0.0.0.0:8000
Performing system checks...
System check identified no issues (0 silenced).
July 05, 2017 - 07:48:51
Django version 1.11.3, using settings 'helloworld.settings'
Starting development server at http://0.0.0.0:8000/
Quit the server with CONTROL-C.
Invalid HTTP_HOST header: '123.206.102.238:8000'. You may need to add u'123.206.102.238' to ALLOWED_HOSTS.
[05/Jul/2017 07:48:58] "GET / HTTP/1.1" 400 61159
Invalid HTTP_HOST header: '123.206.102.238:8000'. You may need to add u'123.206.102.238' to ALLOWED_HOSTS.
[05/Jul/2017 07:48:58] "GET /favicon.ico HTTP/1.1" 400 61108
^\[1] 4615 quit (core dumped) /usr/local/bin/python manage.py runserver 0.0.0.0:8000
去创建的项目中修改 setting.py 文件:
ALLOWED_HOSTS = ['*'] #在这里请求的host添加了*
- 在浏览器输入你服务器的ip及端口号,如果正常启动,输出结果如下:
It worked!
Congratulations on your first Django-powered page.
本文章以下所有列子组织结构
为了方便学习代码上传,github地址
# root @ VM_33_19_centos in /data/PyLearn [15:11:27]
$ tree
.
|-- db.sqlite3
|-- manage.py
|-- PyLearn
| |-- form.py
| |-- form.pyc
| |-- __init__.py
| |-- __init__.pyc
| |-- settings.py
| |-- settings.pyc
| |-- urls.py
| |-- urls.pyc
| |-- viewdb.py
| |-- viewdb.pyc
| |-- view.py
| |-- view.pyc
| |-- wsgi.py
| `-- wsgi.pyc
|-- templates
| |-- base.html
| |-- children.html
| |-- get_form.html
| |-- hello.html
| `-- post_form.html
`-- TestModel
|-- admin.py
|-- admin.pyc
|-- apps.py
|-- __init__.py
|-- __init__.pyc
|-- migrations
| |-- 0001_initial.py
| |-- 0001_initial.pyc
| |-- __init__.py
| `-- __init__.pyc
|-- models.py
|-- models.pyc
|-- tests.py
`-- views.py
4 directories, 34 files
3 编程实例
以下是一个完整的(路由--控制器--视图)实例
路由文件:urls.py
from django.conf.urls import url
from . import view
urlpatterns = [
url(r'^$', view.hello),
]
控制器:view.py
from django.http import HttpResponse
from django.shortcuts import render
def hello(request):
context = {}
context['hello'] = 'Hello World!'
return render(request, 'hello.html', context)
视图文件:hello.html
<h1>{{ hello }}</h1>
配置文件:settings.py
TEMPLATES = [
{
'BACKEND': 'django.template.backends.django.DjangoTemplates',
'DIRS': [BASE_DIR+"/templates"], //绑定视图文件位置
'APP_DIRS': True,
'OPTIONS': {
'context_processors': [
'django.template.context_processors.debug',
'django.template.context_processors.request',
'django.contrib.auth.context_processors.auth',
'django.contrib.messages.context_processors.messages',
],
},
},
]
4 Django 模板标签
4.1 if标签
基本用法
{% if condition %}
... display
{% endif %}
或者:
{% if condition1 %}
... display 1
{% elif condition2 %}
... display 2
{% else %}
... display 3
{% endif %}
{% if %} 标签接受 and , or 或者 not 关键字来对多个变量做判断 ,或者对变量取反( not ),例如:
{% if athlete_list and coach_list %}
athletes 和 coaches 变量都是可用的。
{% endif %}
4.2 for 标签
<ul>
{% for athlete in athlete_list %}
<li>{{ athlete.name }}</li>
{% endfor %}
</ul>
给标签增加一个 reversed 使得该列表被反向迭代:
{% for athlete in athlete_list reversed %}
...
{% endfor %}
可以嵌套使用 {% for %} 标签:
{% for athlete in athlete_list %}
<h1>{{ athlete.name }}</h1>
<ul>
{% for sport in athlete.sports_played %}
<li>{{ sport }}</li>
{% endfor %}
</ul>
{% endfor %}
4.3 ifequal/ifnotequal 标签
{% ifequal %} 标签比较两个值,当他们相等时,显示在 {% ifequal %} 和 {% endifequal %} 之中所有的值。
下面的例子比较两个模板变量 user 和 currentuser :
{% ifequal user currentuser %}
<h1>Welcome!</h1>
{% endifequal %}
和 {% if %} 类似, {% ifequal %} 支持可选的 {% else%} 标签:8
{% ifequal section 'sitenews' %}
<h1>Site News</h1>
{% else %}
<h1>No News Here</h1>
{% endifequal %}
4.4 过滤器
模板过滤器可以在变量被显示前修改它,过滤器使用管道字符,如下所示:
{{ name|lower }}
{{ name }} 变量被过滤器 lower 处理后,文档大写转换文本为小写。
过滤管道可以被* 套接* ,既是说,一个过滤器管道的输出又可以作为下一个管道的输入:
{{ my_list|first|upper }}
以上实例将第一个元素并将其转化为大写。
有些过滤器有参数。 过滤器的参数跟随冒号之后并且总是以双引号包含。 例如:
{{ bio|truncatewords:"30" }}
这个将显示变量 bio 的前30个词。
4.5 include 标签
{% include %} 标签允许在模板中包含其它的模板的内容。
下面这个例子都包含了 nav.html 模板:
{% include "nav.html" %}
4.6 模板继承
接下来我们先创建之前项目的 templates 目录中添加 base.html 文件,代码如下:
templates/base.html 文件代码:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>base</title>
</head>
<body>
<h1>Hello World!</h1>
<p>base.html 段落base。</p>
{% block mainbody %}
<p>base.html 内容base</p>
{% endblock %}
</body>
</html>
以上代码中,名为 mainbody 的 block 标签是可以被继承者们替换掉的部分。
hello.html 中继承 base.html,并替换特定 block,hello.html 修改后的代码如下:
templates/hello.html 文件代码:
{% extends "base.html" %}
{% block mainbody %}
<p>children.html 内容children</p>
{% endblock %}
重新访问地址 http://127.0.0.1:8000/hello,输出结果如下:
Hello World!
base.html 段落base。
children.html 内容children
5 数据模型
5.1 安装mysql数据驱动
pip install mysqlclient
我这里报错 mysql_config: command not found
解决方法是
yum install mysql-devel
Installed:
MariaDB-devel.x86_64 0:10.0.20-1.el7.centos
Complete!
【注】因为我安装的是MariaDB所以这里是MariaDB-devel
5.2 数据库配置
settings.py: 文件代码:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.mysql', # 或者使用 mysql.connector.django
'NAME': 'pylearn',
'USER': 'root',
'PASSWORD': '123456',
'HOST':'localhost',
'PORT':'3306',
}
}
这里添加了中文注释,所以你需要在 HelloWorld/settings.py 文件头部添加 # encoding: utf-8
5.3 定义模型
Django规定,如果要使用模型,必须要创建一个app。
django-admin.py startapp TestModel
目录结构如下:
.
|-- db.sqlite3
|-- manage.py
|-- PyLearn
| |-- __init__.py
| |-- __init__.pyc
| |-- settings.py
|-- templates
| |-- base.html
| |-- children.html
| `-- hello.html
`-- TestModel
|-- admin.py
|-- admin.pyc
|-- apps.py
TestModel/models.py: 文件代码:
# models.py
from django.db import models
class Test(models.Model):
name = models.CharField(max_length=20)
以上的类名代表了数据库表名,且继承了models.Model,类里面的字段代表数据表中的字段(name),数据类型则由CharField(相当于varchar)、DateField(相当于datetime), max_length 参数限定长度。
接下来在settings.py中找到INSTALLED_APPS这一项,如下:
INSTALLED_APPS = (
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'TestModel', # 添加此项
)
在命令行中运行:
python manage.py migrate # 创建表结构
$ python manage.py makemigrations TestModel # 让 Django 知道我们在我们的模型有一些变更
$ python manage.py migrate TestModel # 创建表结构
看到几行 "Creating table…" 的字样,你的数据表就创建好了。
Creating tables ...
……
Creating table TestModel_test #我们自定义的表
……
- 表名组成结构为:应用名_类名(如:TestModel_test)。
- 注意:尽管我们没有在models给表设置主键,但是Django会自动添加一个id作为主键。
5.4 数据库操作
5.4.1 添加数据
urls.py
from django.conf.urls import url
from . import view
from . import viewdb # 新文件
urlpatterns = [
url(r'^$', view.hello),
url(r'^test/$', view.test),
url(r'^testdb/$', viewdb.testdb), # 新路由
]
viewdb.py
$ vim viewdb.py
# -*- coding: utf-8 -*-
from django.http import HttpResponse
from django.shortcuts import render
from TestModel.models import Test
def testdb(request):
test1 = Test(name='archer')
test1.save()
return HttpResponse('<p>添加数据成功</p>')
5.4.2 获取数据
Django提供了多种方式来获取数据库的内容,如下代码所示:
testdb.py: 文件代码:
# -*- coding: utf-8 -*-
from django.http import HttpResponse
from TestModel.models import Test
# 数据库操作
def testdb(request):
# 初始化
response = ""
response1 = ""
# 通过objects这个模型管理器的all()获得所有数据行,相当于SQL中的SELECT * FROM
list = Test.objects.all()
# filter相当于SQL中的WHERE,可设置条件过滤结果
response2 = Test.objects.filter(id=1)
# 获取单个对象
response3 = Test.objects.get(id=1)
# 限制返回的数据 相当于 SQL 中的 OFFSET 0 LIMIT 2;
Test.objects.order_by('name')[0:2]
#数据排序
Test.objects.order_by("id")
# 上面的方法可以连锁使用
Test.objects.filter(name="runoob").order_by("id")
# 输出所有数据
for var in list:
response1 += var.name + " "
response = response1
return HttpResponse("<p>" + response + "</p>")
5.4.3 更新数据
修改数据可以使用 save() 或 update():
testdb.py: 文件代码:
# -*- coding: utf-8 -*-
from django.http import HttpResponse
from TestModel.models import Test
# 数据库操作
def testdb(request):
# 修改其中一个id=1的name字段,再save,相当于SQL中的UPDATE
test1 = Test.objects.get(id=1)
test1.name = 'Google'
test1.save()
# 另外一种方式
#Test.objects.filter(id=1).update(name='Google')
# 修改所有的列
# Test.objects.all().update(name='Google')
return HttpResponse("<p>修改成功</p>")
5.4.4 删除数据
删除数据库中的对象只需调用该对象的delete()方法即可:
testdb.py: 文件代码:
# -*- coding: utf-8 -*-
from django.http import HttpResponse
from TestModel.models import Test
# 数据库操作
def testdb(request):
# 删除id=1的数据
test1 = Test.objects.get(id=1)
test1.delete()
# 另外一种方式
# Test.objects.filter(id=1).delete()
# 删除所有数据
# Test.objects.all().delete()
return HttpResponse("<p>删除成功</p>")
6 表单
urls.py
from django.conf.urls import url
from . import view
from . import viewdb
from . import form
urlpatterns = [
url(r'^$', view.hello),
url(r'^test/$', view.test),
url(r'^testdb/$', viewdb.testdb),
url(r'^get-form/$', form.get_form),
url(r'^submit-get/$', form.submit_get),
url(r'^post-form/$', form.post_form),
url(r'^submit-post/$', form.submit_post),
]
form.py
# -*- coding: utf-8 -*-
from django.http import HttpResponse
from django.shortcuts import render
from django.views.decorators import csrf
def get_form(request):
return render(request, 'get_form.html')
def submit_get(request):
request.encoding='utf-8'
if 'q' in request.GET:
message = '你搜索的内容为: ' + request.GET['q'].encode("utf-8")
else:
message = '你提交了空表单'
return HttpResponse(message)
def post_form(request):
return render(request, 'post_form.html')
def submit_post(request):
ctx ={}
if request.POST:
ctx['rlt'] = request.POST['q']
return render(request, "post_form.html", ctx)
get_form.html
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>get 表单</title>
</head>
<body>
<form action="/submit-get" method="get">
<input type="text" name="q">
<input type="submit" value="搜索">
</form>
</body>
</html>
post_form.html
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>post 表单</title>
</head>
<body>
<form action="/submit-post/" method="post">
{% csrf_token %}
<input type="text" name="q">
<input type="submit" value="Submit">
</form>
<p>{{ rlt }}</p>
</body>
</html>
提交post表单的时候报错
RuntimeError: You called this URL via POST, but the URL doesn’t end in a slash and you have APPEND_SLASH set.
提示form的action地址最后不是/结尾的,而且APPEND_SLASH的值是Ture
将from的action地址改为/结尾的就可以了
或者
修改settings:APPEND_SLASH=False
Django基本使用的更多相关文章
- 异步任务队列Celery在Django中的使用
前段时间在Django Web平台开发中,碰到一些请求执行的任务时间较长(几分钟),为了加快用户的响应时间,因此决定采用异步任务的方式在后台执行这些任务.在同事的指引下接触了Celery这个异步任务队 ...
- 《Django By Example》第四章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:祝大家新年快乐,这次带来<D ...
- django server之间通过remote user 相互调用
首先,场景是这样的:存在两个django web应用,并且两个应用存在一定的联系.某些情况下彼此需要获取对方的数据. 但是我们的应用肯经都会有对应的鉴权机制.不会让人家随随便便就访问的对吧.好比上车要 ...
- Mysql事务探索及其在Django中的实践(二)
继上一篇<Mysql事务探索及其在Django中的实践(一)>交代完问题的背景和Mysql事务基础后,这一篇主要想介绍一下事务在Django中的使用以及实际应用给我们带来的效率提升. 首先 ...
- Mysql事务探索及其在Django中的实践(一)
前言 很早就有想开始写博客的想法,一方面是对自己近期所学知识的一些总结.沉淀,方便以后对过去的知识进行梳理.追溯,一方面也希望能通过博客来认识更多相同技术圈的朋友.所幸近期通过了博客园的申请,那么今天 ...
- 《Django By Example》第三章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:第三章滚烫出炉,大家请不要吐槽文中 ...
- 《Django By Example》第二章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:翻译完第一章后,发现翻译第二章的速 ...
- 《Django By Example》第一章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:本人目前在杭州某家互联网公司工作, ...
- Django
一.Django 简介 Django 是一个由 Python 写成的开放源代码的 Web 应用框架.它最初是被开发来用于管理劳伦斯出版集团旗下的一些以新闻内容为主的网站的,即是 CMS(内容管理系统) ...
- Django admin定制化,User字段扩展[原创]
前言 参考上篇博文,我们利用了OneToOneField的方式使用了django自带的user,http://www.cnblogs.com/caseast/p/5909248.html , 但这么用 ...
随机推荐
- polyfill 一个解决兼容的绝佳方案
polyfill为何物 Polyfill你可以理解为“腻子”,就是装修的时候,可以把缺损的地方填充抹平. 举个例子,html5的storage(session,local), 不同浏览器,不同版本,有 ...
- 如何处理用代码创建SD Sales order时遇到的错误消息KI 180
错误消息KI 180:You must enter a company code for transaction Create sales document 代码: REPORT zcreate_so ...
- chromedp下载文件的方法,备忘一下。
sect := `//a[@href="v/443.json"]` wd,_ := os.Getwd() fmt.Println(wd) return chromedp.Tasks ...
- 【BZOJ4025】二分图(LCT动态维护图连通性)
点此看题面 大致题意: 给你一张图以及每条边的出现时间和消失时间,让你求每个时间段这张图是否是二分图. 二分图性质 二分图有一个比较简单的性质,即二分图中不存在奇环. 于是题目就变成了:让你求每个时间 ...
- activity 工作流学习(一)
一.了解工作流 1.工作流(Workflow),就是“业务过程的部分或整体在计算机应用环境下的自动化”,它主要解决的是“使在多个参与者之间按照某种预定义的规则传递文档.信息或任务的过程自动进行,从而实 ...
- 2018.8.1 Java中的反射和同步详解
为何要使用同步? java允许多线程并发控制,当多个线程同时操作一个可共享的资源变量时(如数据的增删改查), 将会导致数据不准确,相互之间产生冲突,因此加入同步锁以避免在该线程没有完成操作之前,被其他 ...
- 2017.10.21 Java中的数据源与连接池技术
1.数据源技术就是预先建立好一定的数量的数据库连接,并将这些连接保存在连接池中,有连接池负责对这些数据库连接管理,当访问数据库时,只需要从连接池中取出有空闲状态的数据库连接:当程序访问数据库结束时,释 ...
- Meshlab
打开ply文件的软件,Meshlab. 下载 http://yunpan.cn/cgapukD2La9Se (提取码:37f1) http://pan.baidu.com/s/1pJLnWqJ
- Spring/Spring boot中静态变量赋值
情形1:静态变量为自动注入的对象 解决方案:设置两个变量,非静态变量使用@resource注入Bean,然后使用@PostConstruct在Spring初始化Bean成功后为静态变量赋值 @Comp ...
- 理解 JavaScript 作用域(转)
简介 JavaScript 有个特性称为作用域.尽管对于很多开发新手来说,作用域的概念不容易理解,我会尽可能地从最简单的角度向你解释它们.理解作用域能让你编写更优雅.错误更少的代码,并能帮助你实现强大 ...