Django新手十个开发指导
下面是关于Django新手开发中的一些建议,大家可以参考一下~~
1,不要将项目名称包含在引用代码里
比如你创建了一个名为"project"的项目,包含一个名为"app"的应用,那么如下代码是不好的:
from project.app.models import Author
from project.app.models import Author
缺点在于:应用和项目变成了紧耦合,无法将应用轻易变得可重用。如果将来要换一个项目名称,那你可有得受了。
推荐的做法是:
from app.models import Author
from app.models import Author
请注意,你需要将项目的路径配置在PYTHONPATH中。
2,不要硬编码MEDIA_ROOT和TEMPLATE_DIRS
项目配置文件settings.py中不要使用如下代码:
TEMPLATE_DIRS = ( "/home/html/project/templates",)
MEDIA_ROOT = "/home/html/project/appmedia/"
TEMPLATE_DIRS = ( "/home/html/project/templates",)
MEDIA_ROOT = "/home/html/project/appmedia/"
当你在部署到生产环境,或者迁移服务器的时候,就会发生问题。
推荐使用如下方式:
SITE_ROOT = os.path.realpath(os.path.dirname(__file__))
MEDIA_ROOT = os.path.join(SITE_ROOT, 'appmedia')
TEMPLATE_DIRS = ( os.path.join(SITE_ROOT, 'templates'),)
SITE_ROOT = os.path.realpath(os.path.dirname(__file__))
MEDIA_ROOT = os.path.join(SITE_ROOT, 'appmedia')
TEMPLATE_DIRS = ( os.path.join(SITE_ROOT, 'templates'),)
(也可以使用abspath,跟realpath的区别请参考http://rob.cogit8.org/blog/2009/May/05/django-and-relativity-updated/ )
3,不要将静态文件的路径硬编码在模板中
模板中链接CSS,javascript或图片的时候,不建议使用如下方式:
Html代码
<link rel="stylesheet" type="text/css" href="/appmedia/amazing.css" />
<script type="text/javascript" src="/appmedia/jquery.min.js"></script>
<link rel="stylesheet" type="text/css" href="/appmedia/amazing.css" />
<script type="text/javascript" src="/appmedia/jquery.min.js"></script>
当你的项目需要将静态文件用其他服务器提供的时候,通常会是另外一个http地址,那么你就得把所有的/appmedia/替换成新的地址。 没有后顾之忧的解决方法是使用{{ MEDIA_URL }}代替硬编码的路径:
Html代码
<link rel="stylesheet" type="text/css" href="{{ MEDIA_URL }}amazing.css" />
<script type="text/javascript" src="{{ MEDIA_URL }}jquery.min.js"></script>
<link rel="stylesheet" type="text/css" href="{{ MEDIA_URL }}amazing.css" />
<script type="text/javascript" src="{{ MEDIA_URL }}jquery.min.js"></script>
模板上下文变量怎么获取到呢?请使用RequestContext即可:
Python代码
return render_to_response("app/template.html", {'var': 'foo'},
context_instance=RequestContext(request))
return render_to_response("app/template.html", {'var': 'foo'},
context_instance=RequestContext(request))
从RequestContext里还可以获取到当前用户等信息,更详细的介绍请参考:http://www.b-list.org/weblog/2006/jun/14/django-tips-template-context-processors/
4,不要将业务逻辑代码写到视图里
不要迷惑,虽然你可能看过很多书和例子,它们把逻辑都写在了views.py里,但请你别这么做。因为这样不利于单元测试,不利于重用代码。 那我的业务逻辑应该放哪里呢?推荐放到模型里或者单独建立一个辅助(helper)模块。 当然,从模型得到一个Author,获取Author列表的代码是可以放到视图里面的。
5,部署时别忘记将DEBUG设置成False
我们常常忘记在部署时禁用DEBUG,有很多种方法自动来处理这个配置:
import socket if socket.gethostname() == 'productionserver.com':
DEBUG = False
else:
DEBUG = True
import socket
if socket.gethostname() == 'productionserver.com':
DEBUG = False
else:
DEBUG = True
另一种途径是使用不同的配置文件:
Python代码
#文件名:settings_debuy.py
#包含调试模式的配置信息
#使用python manage.py runserver settings=settings_debug.py来运行项目 from settings import * DEBUG = True #还可以配置更多在调试时使用的变量:)
#文件名:settings_debuy.py
#包含调试模式的配置信息
#使用python manage.py runserver settings=settings_debug.py来运行项目
from settings import *
DEBUG = True
#还可以配置更多在调试时使用的变量:)
6,只加载一次自定义的模板标签
当需要使用自定义或者第三方的模板标签和模板过滤器时,通常要在模板中使用:
{% load template_tags %}
{% load template_tags %}
实际情况是,需要在所有用到自定义模板标签和模板过滤器的模板中都使用上面的代码,这样就不DRY了。
Python代码
from django import template
template.add_to_builtins('app.templatetags.custom_tag_module')
from django import template
template.add_to_builtins('app.templatetags.custom_tag_module')
请将以上代码放到项目启动时能加载的模块中(settings.py, urls.py, models.py等)即可。 上面代码的作用是在项目启动时就把自定义模板标签或过滤器加载进来,模板中任何一个地方都可以使用它们,而不需要{% load template_tags %}。
7,合理配置和使用URL
不要将URL全都配置在一个urls.py文件中,比如:
urlpatterns = patterns('',
url(r'^askalumini/question/$','.....registerInstitution',name='iregister'),
url(r'^askalumin/answer/$','someview.....',name='newmemberurl'),
url(r'^institution/member/$','someview.....',name="dashboardurl"),
url(r'^institution/faculty/$','editInstitute',name="editinstituteurl"),
url(r'^memeber/editprofile/$','editProfile',name="editprofileurl"),
url(r'^member/changepassword/$','changePassword',name="changepasswordurl"),
url(r'^member/forgotpassword/$','forgotPassword',name="forgotpasswordurl"),
url(r'^member/changepicture/$','changePicture',name="changepictureurl"),
url(r'^member/logout/$','memeberlogout',name="logouturl"), ,
)
urlpatterns = patterns('',
url(r'^askalumini/question/$','.....registerInstitution',name='iregister'),
url(r'^askalumin/answer/$','someview.....',name='newmemberurl'),
url(r'^institution/member/$','someview.....',name="dashboardurl"),
url(r'^institution/faculty/$','editInstitute',name="editinstituteurl"),
url(r'^memeber/editprofile/$','editProfile',name="editprofileurl"),
url(r'^member/changepassword/$','changePassword',name="changepasswordurl"),
url(r'^member/forgotpassword/$','forgotPassword',name="forgotpasswordurl"),
url(r'^member/changepicture/$','changePicture',name="changepictureurl"),
url(r'^member/logout/$','memeberlogout',name="logouturl"), ,
)
建议的方式是将各应用的URL配置在各自的urls.py中,这样可以使应用更容易重复使用到不同项目里:
urlpatterns = patterns('',
(r'^$', include('institution.urls')),
(r'^institution/', include('institution.urls')),
(r'^askalumini/', include('askalumini.urls')),
(r'^member/', include('member.urls')),
)
urlpatterns = patterns('',
(r'^$', include('institution.urls')),
(r'^institution/', include('institution.urls')),
(r'^askalumini/', include('askalumini.urls')),
(r'^member/', include('member.urls')),
)
如下是应用askalumini的urls.py:
urlpatterns = patterns('askalumini.views',
url(r'^$','askHome',name='askaluminiurl'),
url(r'^questions/(?P<questionno>\d+)/$','displayQuestion',name='askquestiondisplay'),
url(r'^askquestions/$','askQuestion',name='askquestionurl'),
url(r'^postcomment/$','postComment',name="askquestioncomment")
)
urlpatterns = patterns('askalumini.views',
url(r'^$','askHome',name='askaluminiurl'),
url(r'^questions/(?P<questionno>\d+)/$','displayQuestion',name='askquestiondisplay'),
url(r'^askquestions/$','askQuestion',name='askquestionurl'),
url(r'^postcomment/$','postComment',name="askquestioncomment")
)
刚才提到静态文件路径不要硬编码,url的处理方式也尽量不要硬编码,否则当你更改一个地址时会牵涉到多处的修改,可以使用一些url函数来处理。
在/project/askalumini/urls.py中,为每一个url定义了name,它可以帮助我们有效地在视图、模板和模型中处理url,而不是硬编码。 为保证名称的唯一,请遵照将url命名为<appname>/<somelabel>的习惯用法。
举例来说,在views.py文件中有如下代码:
Python代码
HttpResponseRedirect("/askalumini/questions/54")
HttpResponseRedirect("/askalumini/questions/54")
请改为:
from django.core.urlresolvers import reverse
HttpResponseRedirect(reverse('askquestiondisplay',kwargs={'questionno':q.id}))
from django.core.urlresolvers import reverse
HttpResponseRedirect(reverse('askquestiondisplay',kwargs={'questionno':q.id}))
在模型中使用models.permalink装饰器来格式url:
Python代码
@models.permalink
def get_absolute_url(self):
return ('profileurl2',(),{'userid': self.user.id})
@models.permalink
def get_absolute_url(self):
return ('profileurl2',(),{'userid': self.user.id})
在模板中使用url标签代替硬编码:
Html代码
{% url askquestiondisplay 345 %}
<a href="{% url askquestiondisplay 345 %}"> Ask Question </a>
{% url askquestiondisplay 345 %}
<a href="{% url askquestiondisplay 345 %}"> Ask Question </a>
8,调试
调试通常会借助一些第三方工具来获得更多的运行时信息。 一个请求执行了多少句SQL?花了多长时间? 调用的哪个模板?客户端设置了什么COOKIE?SESSION呢?。。。
你可以使用django-debug-toolbar查看上面甚至更多的信息:http://github.com/robhudson/django-debug-toolbar
另一个工具是Werkzeug debugger,它可以在错误页面打开python shell,让你更方便的跟踪错误信息,请访问:http://blog.dpeepul.com/2009/07/14/python-shell-right-on-the-django-error-page/ 获得更多信息。
还有pdb,一个强大的调试工具:http://ericholscher.com/blog/2008/aug/31/using-pdb-python-debugger-django-debugging-series-/
9,了解pinax备用
django最大的优点是代码重用,DRY,pinax就是这样一个平台,包含了许多可拿来直接使用的代码,比如openid,电子邮件验证等等。请访问:http://pinaxproject.com/
10,了解一些著名的第三方应用
1)数据库升级工具
什么是数据库升级工具?你运行了syncdb,运行了一年之后,对模型做了更改,添加了字段,删除了字段,要再运行syncdb吗?或者ALTER TABLE ...?
django-evolutions可以帮你完成上面的事情,但它好像不够强壮:http://code.google.com/p/django-evolution/
South能很强壮地完成上面的事情,但是需要学学怎么用:http://south.aeracode.org/
2)模板系统
django自带的模板系统是可以替换的,并且各自有优缺点。
template-utils增强了模板的比较标签等功能 ,并提供其他的一些实用特性:http://django-template-utils.googlecode.com/svn/trunk/docs/
Jinja是一个完整的第三方模板系统,可以替换默认模板系统,它提供了许多优越的特性:http://jinja.pocoo.org/2/
3)第三方应用
django command extensions提供了很多实用的命令行功能:
shell_plus加载所有django模型
runserver_plus整合了Werkzeug调试工具
生成模型图表,你可以展示给你的老板 …… 请参考:http://ericholscher.com/blog/2008/sep/12/screencast-django-command-extensions/
Sorl可以生成缩略图:http://code.google.com/p/sorl-thumbnail/
END
以上内容转自http://shinyzhu.iteye.com/blog/593427
Django新手十个开发指导的更多相关文章
- Django新手图文教程
Django新手图文教程 本文面向:有python基础,刚接触web框架的初学者. 环境:windows7 python3.5.1 pycharm专业版 Django 1.10版 pip3 一 ...
- pycharm+python+Django之web开发环境的搭建(windows)
转载:https://blog.csdn.net/yjx2323999451/article/details/53200243/ pycharm+python+Django之web开发环境的搭建(wi ...
- Django 新手图文教程 (转)
简约而不简单的 Django 新手图文教程 环境:windows7,python3.5.1,pycharm专业版,Django 1.10版,pip3 一.Django简介 百度百科:开放源代码的Web ...
- Django 博客开发教程目录索引
Django 博客开发教程目录索引 本项目适合 0 基础的 Django 开发新人. 项目演示地址:Black & White,代码 GitHub 仓库地址:zmrenwu/django-bl ...
- Django高级实战 开发企业级问答网站
Django高级实战 开发企业级问答网站 整个课程都看完了,这个课程的分享可以往下看,下面有链接,之前做java开发也做了一些年头,也分享下自己看这个视频的感受,单论单个知识点课程本身没问题,大家看的 ...
- [Python] 利用Django进行Web开发系列(一)
1 写在前面 在没有接触互联网这个行业的时候,我就一直很好奇网站是怎么构建的.现在虽然从事互联网相关的工作,但是也一直没有接触过Web开发之类的东西,但是兴趣终归还是要有的,而且是需要自己动手去实践的 ...
- [Python] 利用Django进行Web开发系列(二)
1 编写第一个静态页面——Hello world页面 在上一篇博客<[Python] 利用Django进行Web开发系列(一)>中,我们创建了自己的目录mysite. Step1:创建视图 ...
- Nginx+Python+uwsgi+Django的web开发环境安装及配置
Nginx+Python+uwsgi+Django的web开发环境安装及配置 nginx安装 nginx的安装这里就略过了... python安装 通常系统已经自带了,这里也略过 uwsgi安装 官网 ...
- Django 入门案例开发(上)
Django 入门案例开发(中) http://www.cnblogs.com/focusBI/p/7858267.html Django是一个重量级的web开发框架,它提供了很多内部已开发好的插件供 ...
随机推荐
- 多线程学习笔记(四)---- Thread类的其他方法介绍
一.wait和 sleep的区别 wait可以指定时间也可以不指定时间,而sleep必须指定时间: 在同步中时,对cpu的执行权和锁的处理不同: wait:释放执行权,释放锁:释放锁是为了别人noti ...
- springmvc <mvc:resource /> 标签使用
<!-- 配置静态资源 --><mvc:resources location="/static/" mapping="/static/**"/ ...
- 阿里云服务器扩展分区和文件系统_Linux数据盘
官方文档永远是最好的 https://help.aliyun.com/document_detail/25452.html?spm=a2c4g.11186623.6.786.5fde4656Ln6AO ...
- git rebase解决合并冲突
git rebase解决合并冲突 记录合并冲突解决方法,使用的git rebase,感觉很好用 1.git rebase 文档 https://git-scm.com/docs/git-rebas ...
- mongodb 指令
db.xxx.stats() 查看表的大小 db.xxx.remove({'endtime':{'$lte':ISODate('2018-10-01')}}) 删除小于等于固定时间的数据. db.us ...
- 2017蓝桥杯日期问题(C++B组)
标题:日期问题小明正在整理一批历史文献.这些历史文献中出现了很多日期.小明知道这些日期都在1960年1月1日至2059年12月31日.令小明头疼的是,这些日期采用的格式非常不统一,有采用年/月/日的, ...
- Array(数组)对象-->数组的访问
1.访问数组: 通过指定数组名以及索引号码,你可以访问某个特定的元素. 格式: 数组对象名[下标] 例如:arr[0] 就是访问数组第一个值 var arr = new Array(3); arr[ ...
- alg-链表中有环
typedef struct ListNode { int val; ListNode *next; ListNode(int x) : val(x), next(nullptr) {} }ListN ...
- Netty是如何处理新连接接入事件的?
更多技术分享可关注我 前言 前面的分析从Netty服务端启动过程入手,一路走到了Netty的心脏——NioEventLoop,又总结了Netty的异步API和设计原理,现在回到Netty服务端本身,看 ...
- 安卓广播api介绍,给自己理清楚概念
广播接收器类概述 这是用于接收由sendBroadcast()发送intent的基类.这个类一般都会被继承重写里面的onReceive()方法..如果您不需要跨应用程序发送广播,请考虑使用LocalB ...