[TOC]
1.粗略统计
博客阅读数的统计根据业务的精细度可以多种方法实现,其中最简单的方式就是在models.py定义readed_num字段,然后在打开文章详情页时候给阅读数字段+1即可(此方法计算不准确,刷新页面也会使得阅读数增加,在粗略计算下可以使用)
2.使用cookie统计
设置cookie,并获取cookie来判断该篇文章是否阅读过
根据设置访问文章的cookie的键key和值value来标记文章是否被阅读过,如果阅读过设置cookie,在打开文章时候先获取cookie来判断是否存在,如果不存在,阅读数+1;cookie存在则阅读数保持不变
存在问题:
1.每次阅读数改变的时候,admin管理后台的更新时间会跟着变化
2.当管理员在编辑某篇文章,此时有人点击该篇文章,阅读数增加,但是管理员编辑修改完文章保存后,阅读数会是管理员编辑文章时候的阅读数量
def blog_detail(request,blog_pk):
"""博客详情页"""
blog=get_object_or_404(Blog,pk=blog_pk) # 根据传入blog_pk的ID来找到具体对应博客文章
# 获取cookie:根据获取cookie的key值'blog_%s_readed' % blog_pk,判断是否存在,不存在则阅读数+1
if not request.COOKIES.get('blog_%s_readed' % blog_pk):
blog.readed_num += 1 #点击该篇文章,阅读数自增1
blog.save()
pre_blog=Blog.objects.filter(id__gt=blog.id).last() # 博客的上一篇文章
next_blog=Blog.objects.filter(id__lt=blog.id).first() # 博客的下一篇文章
response=render(request,'blog/blog_detail.html',{'blog':blog,'pre_blog':pre_blog,'next_blog':next_blog}) #响应
# 设置cookie,打开过这篇文章即写入cookie,key是'blog_%s_readed' % blog_pk,value是'true'
response.set_cookie('blog_%s_readed' % blog_pk,'true')
return response
3.模型分离
Github源码地址:请点这里
-
1.创建模型
在blog应用下面的models.py添加ReadNum类:
代码示例:
class ReadNum(models.Model):
read_num=models.IntegerField(default=0,verbose_name='阅读数')
blog=models.OneToOneField(Blog,on_delete=models.DO_NOTHING,verbose_name='标题')
class Meta:
verbose_name='阅读数'
verbose_name_plural=verbose_name
这种方式可以避免方式2中存在的2个问题
2.同步数据库操作,模型创建完成后记得:
python manage.py makemigratiosn
python manage.py migrate
-
3.admin.py添加到后台管理页面
class ReadNumAdmin(admin.ModelAdmin):
list_display=('id','read_num','blog')
admin.site.register(ReadNum,ReadNumAdmin)
- 为了让阅读数在blog管理页面显示,在models.py的Blog模型添加get_read_num()方法
注意:此处会如果谋篇文章暂时还没有阅读数,默认情况下显示的不是0,而是-,实际上是抛出了一个异常,该异常不处理,则显示阅读数是0的显示-,处理后正常显示0
错误显示为:
blog应用下面的Blog模型里添加
from django.db.models.fields import exceptions
# 添加方法,添加的read_num方法,用于在admin管理后台的blog页面显示阅读数。即在admin.py的BlogAdmin类的list_display添加read_num
def get_read_num(self):
# 返回该篇文章的阅读数,此处如果该文章还没有阅读数,会有个异常,该异常是的默认阅读数不是0,而是-
try:
return self.readnum.read_num
except exception.ObjectDoesNotExist:
return 0
- 4.views.py逻辑处理
此处需要先判断是否cookie存在,不存在则继续判断该篇文章的阅读数是否存在,如果存在则阅读数+1并保存数据库;如果不存在,需要先创建该篇文章的阅读数(创建是为了在阅读数模型保存数据,具体如下admin管理页面截图),再阅读数+1并保存数据库
def blog_detail(request,blog_pk):
"""博客详情页"""
blog=get_object_or_404(Blog,pk=blog_pk) # 根据传入blog_pk的ID来找到具体对应博客文章
# 获取cookie:根据获取cookie的key值'blog_%s_readed' % blog_pk,判断是否存在,不存在则阅读数+1
if not request.COOKIES.get('blog_%s_read' % blog_pk):
if ReadNum.objects.filter(blog=blog).count():
# 记录存在,则查询并阅读数+1,保存
readnum=ReadNum.objects.get(blog=blog)
else:
# 对应记录不存在,则创建并阅读数+1,保存(这样就会在阅读数管理页面点击后保存对应文章以及阅读数)
readnum=ReadNum(blog=blog)
readnum.read_num += 1 #点击该篇文章,阅读数自增1
readnum.save()
pre_blog=Blog.objects.filter(id__gt=blog.id).last() # 博客的上一篇文章
next_blog=Blog.objects.filter(id__lt=blog.id).first() # 博客的下一篇文章
response=render(request,'blog/blog_detail.html',{'blog':blog,'pre_blog':pre_blog,'next_blog':next_blog}) #响应
# 设置cookie,打开过这篇文章即写入cookie,key是'blog_%s_readed' % blog_pk,value是'true'
response.set_cookie('blog_%s_read' % blog_pk,'true')
return response
-
5.修改html模版文件阅读的显示
直接使用Blog模型中的get_read_num()方法