python官网 中文版 新闻-新闻主页 - python兵者 - 博客园

一、功能需求分析

1、banner

2、推荐文章列表

3、文章标签导航

4、文章列表

5、分页

二、模型设计

根据功能分析,我们需要如下表,大量的经验和功能积累

1、表和字段分析

a 文章分类表

b 文章表

c 文章评论表

d 推荐文章表

e 轮播图表

2、模型定义

定义一个基类模型,抽取公共字段

创建时间、更新时间、逻辑删除

a 分析每个表功能、根据字段编写模型代码

三、文章标签导航功能

1.接口设计

类目          说明

请求方式        GET

url            /

参数说明        无参数

2.返回结果

返回新闻页面,直接在模版渲染

#insert news tag data

INSERT INTO tb_tag(name, create_time, update_time, is_delete) values

('Python基础', now(), now(), 0),

('Python高级', now(), now(), 0),

('Python函数', now(), now(), 0),

('PythonGUI', now(), now(), 0),

('Linux教程', now(), now(), 0),

('Python框架', now(), now(), 0);

四、新闻列表功能

1、业务流程分析

a 判断前端传递标签分类ID是否为空,是否为整数,是否超过范围

b 判断前端传递当前文章页数是否为空,是否为整数,是否超过范围

2、接口设计

类目          说明

请求方式        GET

url           /news/

参数说明:

参数名    类型    是否必须      描述

tag      整数      否        标签分类ID

page     整数      是        当前文章页数

3、返回结果:

json

{

"errno": "0",

"errmsg": "",

"data": {

"total_pages": 61,

"news": [

{

"digest": "在python用import或者from...import或者from...import...as...来导入相应的模块,作用和使用方法与C语言的include头文件类似。其实就是引入...",

"title": "import方法引入模块详解",

"author": "python",

"image_url": "/media/jichujiaochen.jpeg",

"tag_name": "Python基础",

"update_time": "2018年12月17日 14:48"

},

{

"digest": "如果你原来是一个php程序员,你对于php函数非常了解(PS:站长原来就是一个php程序员),但是现在由于工作或者其他原因要学习python,但是p...",

"title": "给曾经是phper的程序员推荐个学习网站",

"author": "python",

"image_url": "/media/jichujiaochen.jpeg",

"tag_name": "Python基础",

"update_time": "2018年12月17日 14:48"

}

]

}

}

mysql -u root -p -D tzpj < tb_news_20181217.sql. 导入数据库

news/modile.py代码

from django.db import models

from utils.models import BaseModel

class Tag(BaseModel):

"""

文章分类标签模型

"""

# 字段

name = models.CharField('标签名', max_length=64, help_text='标签名')

class Meta:

ordering = ['-update_time', '-id'] #排序

db_table = "tb_tag" #指明数据库表名

verbose_name = '文章标签' #在admin站点中显示点名称

verbose_name_plural = verbose_name #显示点复数名称

def __str__(self):

return self.name

class News(BaseModel):

"""

文章模型

"""

title = models.CharField('标题', max_length=150, help_text='标题')

digest = models.CharField('摘要', max_length=200, help_text='摘要')

content = models.TextField('内容', help_text='内容')

clicks = models.IntegerField('点击量', default=0, help_text='点击量')

image_url = models.URLField('图片url', default='', help_text='图片url')

tag = models.ForeignKey('Tag', on_delete=models.SET_NULL, null=True)

author = models.ForeignKey('user.User', on_delete=models.SET_NULL, null=True)

class Meta:

ordering = ['-update_time', '-id'] #排序

db_table = "tb_news" #指明数据库表名

verbose_name = '新闻' #在admin站点中显示点名称

verbose_name_plural = verbose_name #显示点复数名称

def __str__(self):

return self.title

class Comments(BaseModel):

"""

评论模型

"""

content = models.TextField('内容', help_text='内容')

author = models.ForeignKey('user.User', on_delete=models.SET_NULL, null=True)

news = models.ForeignKey('News', on_delete=models.CASCADE)

class Meta:

ordering = ['-update_time', '-id'] #排序

db_table = "tb_comments" #指明数据库表名

verbose_name = '评论' #在admin站点中显示点名称

verbose_name_plural = verbose_name #显示点复数名称

def __str__(self):

return '<评论{}>'.format(self.id)

class HotNews(BaseModel):

"""

推荐文章模型

"""

news = models.OneToOneField('News', on_delete=models.CASCADE)

priority = models.IntegerField('优先级', help_text='优先级')

class Meta:

ordering = ['-update_time', '-id'] #排序

db_table = "tb_hotnews" #指明数据库表名

verbose_name = '热门文章' #在admin站点中显示点名称

verbose_name_plural = verbose_name #显示点复数名称

def __str__(self):

return '<热门文章{}>'.format(self.id)

class Banner(BaseModel):

"""

轮播图

"""

image_url = models.URLField('轮播图', help_text='轮播图url')

priority = models.IntegerField('优先级', help_text='优先级')

news = models.OneToOneField('News', on_delete=models.CASCADE)

class Meta:

ordering = ['priority', '-update_time', '-id'] #排序

db_table = "tb_banner" #指明数据库表名

verbose_name = '热门文章' #在admin站点中显示点名称

verbose_name_plural = verbose_name #显示点复数名称

def __str__(self):

return '<轮播图{}>'.format(self.id)

news/views.py代码

import logging

from django.shortcuts import render

from django.views import View

from django.db.models import F

from django.core.paginator import Paginator

from .models import Tag, News

from . import constants

from utils.res_code import json_response

logger = logging.getLogger('django')

def index(request):

"""

新闻首页视图 only默认带ID

url: /

:param request:

:return:

"""

# 新闻标签

tags = Tag.objects.only('name').filter(is_delete=False)

return render(request, 'news/index.html', context={

'tags': tags

})

class NewsListView(View):

"""

新闻列表视图

"""

def get(self, request):

# 1,获取参数

try:

tag_id = int(request.GET.get('tag', 0))

except Exception as e:

logger.error('标签错误, {}'.format(e))

tag_id = 0

try:

tag_id = int(request.GET.get('tag', 0))

except Exception as e:

logger.error('标签错误, {}'.format(e))

tag_id = 1

# 2,获取查询集 values返回字典

news_queryset = News.objects.values('id', 'title', 'digest','image_url', 'update_time',

'tag__name', 'author__username').annotate(tag_name=F('tag_name'),author=F('author__username'))

#过滤

# if tag_id:

# news = news_queryset.filter(is_dalete=False, tag_id=tag_id)

# else:

# news = news_queryset.filter(is_dalete=False)

news = news_queryset.filter(is_dalete=False, tag_id=tag_id) or news_queryset.filter(is_dalete=False)

# 3,分页

paginator = Paginator(news, constants.PER_PAGE_NEWS_COUNT)

#获取当前页数据 get_page可以容错

news_info = paginator.get_page(page)

# 4,返回数据

data = {

'total_pages': paginator.num_pages,

'news': list(news_info)

}

return json_response(data=data)

序列化:把特定的内存对象转化成可以存储的字符串

utils/res_code.py代码

import datetime

from django.http import JsonResponse

from django.core.serializers.json import DjangoJSONEncoder

class MyJSONEncoder(DjangoJSONEncoder):

def default(self, o):

if isinstance(o, datetime.datetime):

return o.astimezone().strftime('%Y-%m-%d %H:%M:%S') #转换为本地时间

else:

return super().default(o)

def json_response(errno=Code.OK, errmsg='', data=None, kwargs=None):

json_dict = {

'errno': errno,

'errmsg': errmsg,

'data': data

}

if kwargs and isinstance(kwargs, dict):

json_dict.update(kwargs)

return JsonResponse(json_dict, encoder=MyJSONEncoder)

五、推荐新闻

1,接口设计

类目          说明

请求方式        GET

url            /

参数说明        无参数

2,返回结果

返回新闻页面,直接在模版渲染

news/vies.py代码

def index(request):

"""

新闻首页视图

url: /

"""

# 新闻标签

tags = Tag.objects.only('name').filter(is_delete=False)

# 热门新闻 objects.select_related('news')一次性从数据库多拿数据

hot_news = HotNews.objects.select_related('news').only('news__title', 'news__image_url', 'news_id').filter(is_delete=False).order_by('priority', '-news__clicks')[:constants.SHOW_HOTNEWS_COUNT]

return render(request, 'news/index.html', context={

'tags': tags,

'hot_news': hot_news

})

六、轮播图功能

接口设计

1,接口说明:

类目        说明

请求方法      GET

url定义      /news/banners/

参数格式      无参数

2,返回结果: # json

{

'erron': '0',

'errmsg': 'ok'

'data': {

'banners':[

{

'image_url':'/media/jichujiaochen.jpeg',

'news_id': 221,

'news_title': 'python算法快速排序'

},

{

'image_url':'/media/python_advanced.jpeg',

'news_id': 707,

'news_title': 'python序列与映射的解包操作'

}

]

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值