python

准备工作

新建一个django项目

# 新建一个django项目
$ django-admin startproject mysite
# 新建一个app
$ django-admin startapp blog

项目的结构

├── blog
│ ├── admin.py
│ ├── apps.py
│ ├── __init__.py
│ ├── migrations
│ │ └── __init__.py
│ ├── models.py
│ ├── tests.py
│ └── views.py
├── manage.py
└── mysite
├── __init__.py
├── settings.py
├── urls.py
└── wsgi.py# mysite/settings.py
installed_apps = [
‘django.contrib.admin’,
‘django.contrib.auth’,
‘django.contrib.contenttypes’,
‘django.contrib.sessions’,
‘django.contrib.messages’,
‘django.contrib.staticfiles’,
‘blog’,
‘markdown2’
]$ python3 manage.py runserver
$ python manage.py collectstatic

一般在urls.py中配置url,在models.py中配置model,在views.py中配置view。

urls.py

function views

1. add an import: from my_app import views
2. add a url to urlpatterns: url(r’^$’, views.home, name=’home’)

class-based views

1. add an import: from other_app.views import home
2. add a url to urlpatterns: url(r’^$’, home.as_view(), name=’home’)

including another urlconf

1. import the include() function: from django.conf.urls import url, include
2. add a url to urlpatterns: url(r’^blog/’, include(‘blog.urls’))# blog/urls.py
from django.conf.urls import url
from blog import views
urlpatterns = [
url(r’^blog/$’, views.indexview.as_view(), name=’index’),
url(r’^blog/article/(?p\d+)$’, views.articledetailview.as_view(), name=’detail’),
url(r’^blog/category/(?p\d+)$’, views.categoryview.as_view(), name=’category’),
url(r’^blog/tag/(?p\d+)$’, views.tagview.as_view(), name=’tag’),
]

使用(?pd+)的形式捕获值给中得参数,比如(?pd+),当访问/blog/article/3时,将会将3捕获给article_id,这个值会传到views.articledetailview。

# mysite/urls.py
from django.conf.urls import url, include
from django.contrib import admin
from blog import views
urlpatterns = [
url(r’^admin/’, admin.site.urls),
url(r”, include(‘blog.urls’, namespace=’blog’, app_name=’blog’))
]

其中namespace参数为我们指定了命名空间,这说明这个urls.py中的url是blog app下的,这样即使不同的app下有相同url也不会冲突了。

假设用户要访问某篇文章,它会自动解析 blog:detail 这个视图函数对应的 url,并且把 article.pk(文章的主键)传递给detail视图函数,details就是我们在blog/urls.py中指定的name。

{{ article.title }}

如果要访问某个目录

{{ category.name }}models.py

django.db.models是orm框架的基础,在blog/models.py中新建article, category, tag三个model。

class article(models.model):
status_choices = (
(‘d’, ‘draft’),
(‘p’, ‘published’),
)
# 仍然使用默认的 objects 作为 manager 的名字
objects = articlemanager()
title = models.charfield(‘标题’, max_length=70)
body = models.textfield(‘正文’)
created_time = models.datetimefield(‘创建时间’, auto_now_add=true)
last_modified_time = models.datetimefield(‘修改时间’, auto_now=true)
status = models.charfield(‘文章状态’, max_length=1, choices=status_choices)
# blank和null要同时设置为null,详情参考官方文档
abstract = models.charfield(‘摘要’, max_length=54, blank=true, null=true,
help_text=”可选,如若为空将摘取正文的前54个字符”)
views = models.positiveintegerfield(‘浏览量’, default=0)
likes = models.positiveintegerfield(‘点赞数’, default=0)
topped = models.booleanfield(‘置顶’, default=false)
category = models.foreignkey(‘category’, verbose_name=’分类’, null=true, on_delete=models.set_null)
tags = models.manytomanyfield(‘tag’, verbose_name=’标签集合’, blank=true)
def __str__(self):
return self.title
class meta:
ordering = [‘-last_modified_time’]
# 新增 get_absolute_url 方法
def get_absolute_url(self):
# 这里 reverse 解析 blog:detail 视图函数对应的 url
return reverse(‘blog:detail’, kwargs={‘article_id’: self.pk})

django给我们提供了很多有用的字段,比如上面提到的charfiled, testfield, datetimefiled等等,详情请参考官方文档。

django中的一对多是在一中进行设置,这里对应于文章的分类,foreignkey即数据库中的外键。on_delete=models.set_null表示删除某个分类(category)后该分类下所有的article的外键设为null(空),所以我们同时设置了null=true。多对多就不同,两边都要进行配置。详情请参考官方文档。

class category(models.model):
name = models.charfield(‘类名’, max_length=20)
created_time = models.datetimefield(‘创建时间’, auto_now_add=true)
last_modified_time = models.datetimefield(‘修改时间’, auto_now=true)
def __str__(self):
return self.nameclass tag(models.model):
name = models.charfield(‘标签名’, max_length=20)
created_time = models.datetimefield(‘创建时间’, auto_now_add=true)
last_modified_time = models.datetimefield(‘修改时间’, auto_now=true)
def __str__(self):
return self.name

评论功能的实现

class blogcomment(models.model):
user_name = models.charfield(‘评论者名字’, max_length=100)
user_email = models.emailfield(‘评论者邮箱’, max_length=255)
body = models.textfield(‘评论内容’)
created_time = models.datetimefield(‘评论发表时间’, auto_now_add=true)
article = models.foreignkey(‘article’, verbose_name=’评论所属文章’, on_delete=models.cascade)
def __str__(self):
return self.body[:20]class articlemanage(models.manager):
“””
继承自默认的 manager ,为其添加一个自定义的 archive 方法
“””
def archive(self):
date_list = article.objects.datetimes(‘created_time’, ‘month’, order=’desc’)
# 获取到降序排列的精确到月份且已去重的文章发表时间列表
# 并把列表转为一个字典,字典的键为年份,值为该年份下对应的月份列表
date_dict = defaultdict(list)
for d in date_list:
date_dict[d.year].append(d.month)
# 模板不支持defaultdict,因此我们把它转换成一个二级列表,由于字典转换后无序,因此重新降序排序
return sorted(date_dict.items(), reverse=true)

我们首先要在project_name/settings.py中配置好相应的配置文件

databases = {
‘default’: {
‘engine’: ‘django.db.backends.mysql’,
‘name’: ‘db_name’,
‘user’: ‘db_user’,
‘password’: ‘db_password’,
‘host’: ‘localhost’, # or an ip address that your db is hosted on
‘port’: ‘3306’,
}
}

定义完毕后,我们执行下面的命令就在数据库中可以生成相应的数据表:

$ python manage.py makemigrations
$ python manage.py migrateadmins.py

参考mozila的教程以及结合官方文档。

views.py

下面要使用markdown2,所以在installed_app里面要添加markdown2,不过这个mardown解析非常的不好,并且弄完还要去下载相应的markdown的css文件,有个专门的网站。

from blog.models import article, tag, category
from django.views.generic import listview, detailview
import markdown2
class indexview(listview):
# template_name属性用于指定使用哪个模板进行渲染
template_name = “blog/index.html”
# context_object_name属性用于给上下文变量取名(在模板中使用该名字)
context_object_name = “article_list”
def get_queryset(self):
article_list = article.objects.filter(status=’p’)
for article in article_list:
article.body = markdown2.markdown(article.body, )
return article_list
def get_context_data(self, **kwargs):
kwargs[‘category_list’] = category.objects.all().order_by(‘name’)
# 调用 archive 方法,把获取的时间列表插入到 context 上下文中以便在模板中渲染
kwargs[‘date_archive’] = article.objects.archive()
kwargs[‘tag_list’] = tag.objects.all().order_by(‘name’)
return super(indexview, self).get_context_data(**kwargs)

上面因为我们要进行markdown处理,所以重新自定义了get_queryset,如果不要进行相应的处理,直接制定model就行了,get_context_data可以添加一些额外的字段,比如以后我们要在首页的侧边栏显示目录和标签,所以这里要添加一个category_list和tag_list。

class articledetailview(detailview):
model = article
template_name = “blog/detail.html”
context_object_name = “article”
# pk_url_kwarg会自动和model中相应的主键对应,aritlce_id就是下面配置的urlconf
pk_url_kwarg = ‘article_id’
# 为了让文章以markdown形式展现,我们重写get_object()方法
def get_object(self):
obj = super(articledetailview, self).get_object()
obj.body = markdown2.markdown(obj.body)
return obj
# 新增 form 到 context
def get_context_data(self, **kwargs):
kwargs[‘comment_list’] = self.object.blogcomment_set.all()
kwargs[‘form’] = blogcommentform()
return super(articledetailview, self).get_context_data(**kwargs)class categoryview(listview):
template_name = “blog/index.html”
context_object_name = “article_list”
def get_queryset(self):
# url里的cate_id传递给categoryview,传递的参数在kwargs属性中获取
article_list = article.objects.filter(category=self.kwargs[‘cate_id’],status=’p’)
for article in article_list:
article.body = markdown2.markdown(article.body, )
return article_list
def get_context_data(self, **kwargs):
# 增加一个category_list,用于在页面显示所有分类,按照名字排序
kwargs[‘category_list’] = category.objects.all().order_by(‘name’)
return super(categoryview, self).get_context_data(**kwargs)class tagview(listview):
template_name = “blog/index.html”
context_object_name = “article_list”
def get_queryset(self):
“””
根据指定的标签获取该标签下的全部文章
“””
article_list = article.objects.filter(tags=self.kwargs[‘tag_id’], status=’p’)
for article in article_list:
article.body = markdown2.markdown(article.body, extras=[‘fenced-code-blocks’], )
return article_list
def get_context_data(self, **kwargs):
kwargs[‘tag_list’] = tag.objects.all().order_by(‘name’)
return super(tagview, self).get_context_data(**kwargs)from django.views.generic.edit import formview
class commentpostview(formview):
form_class = blogcommentform
template_name = ‘blog/detail.html’
def form_valid(self, form):
target_article = get_object_or_404(article, pk=self.kwargs[‘article_id’])
# 调用modelform的save方法保存评论,设置commit=false则先不保存到数据库,
# 而是返回生成的comment实例,直到真正调用save方法时才保存到数据库。
comment = form.save(commit=false)
# 把评论和文章关联
comment.article = target_article
comment.save()
# 评论生成成功,重定向到被评论的文章页面,get_absolute_url 请看下面的讲解。
self.success_url = target_article.get_absolute_url()
return httpresponseredirect(self.success_url)
def form_invalid(self, form):
target_article = get_object_or_404(article, pk=self.kwargs[‘article_id’])
# 不保存评论,回到原来提交评论的文章详情页面
return render(self.request, ‘blog/detail.html’, {
‘form’: form,
‘article’: target_article,
‘comment_list’: target_article.blogcomment_set.all(),
})template

{% for %}循环标签,{% if %}判断标签. {{ variable }}是一些非常常用的标签

在模板文件中我们可以这样使用,views.py中已经指定了context_object_name = “article_list”,并且已经在get_queryset()中进行了markdown处理

{% for article in article_list %}
{{article.title}}

通常都会设置一个通用的父模板:

{% extends “base_generic.html” %}
{% block content %}

{% endblock %}

好像要这么这么设置:

templates = [
{
‘backend’: ‘django.template.backends.django.djangotemplates’,
‘dirs’: [os.path.join(base_dir, ‘blog/templates’)]
,
‘app_dirs’: true,

]静态文件

由于源代码丢失,具体情况记得不太清晰,静态文件路径要设置好,如果js文件加载异常,可能是加载顺序的问题。

base_generic.html大概就是下面这样的格式:

{% load staticfiles %}

myblog

下面这样设置貌似有点问题:

# mysite/settings.py
static_url = ‘/static/’
staticfiles = os.path.join(base_dir, ‘blog/static’)

具体参考官方文档

部署

使用uwsgi+nginx

/etc/nginx/sites-available/mysite.conf,blog是app名字,static文件放在了下面,建议直接放在mysite下面,template也是一样:

server {
listen 80;
location /static/ {
alias /home/omrsf/mysite/blog/static/;
}
location / {
uwsgi_pass 127.0.0.1:8001;
include /etc/nginx/uwsgi_params;
}
}

uwsgi -i uwsgi.ini来启动uwsgi进程,结合nohup &:

[uwsgi]
socket = 127.0.0.1:8001
chdir=/home/ormsf/mysite/
wsgi-file = mysite/wsgi.py
processes = 2
threads = 4
chmod-socket = 664改进

目前文章是直接在admin.py中注册model,然后去admin后台发布的,可以做成api接口,做一个在线的编辑器。增加基本的用户认证功能。

零碎知识点null和blank的区别

null 是针对数据库而言,如果 null=true, 表示数据库的该字段可以为空。

blank 是针对表单的,如果 blank=true,表示你的表单填写该字段的时候可以不填,比如 admin 界面下增加 model 一条记录的时候。直观的看到就是该字段不是粗体。

render与render_response

优先采用render。

get_absolute_url

model有一个get_absolute_url,它可以与reverse结合起来。

更多python-django相关文章请关注php中文网!

Posted in 未分类

发表评论