美文网首页Django学习Django日记Python开发
Python日记——用Django做一个简易的知乎日报API

Python日记——用Django做一个简易的知乎日报API

作者: 饥渴计科极客杰铿 | 来源:发表于2016-07-16 00:05 被阅读1450次

    为什么我要选择django做生成api的框架
    因为其内置的orm简单易用
    生成restful的api要配合rest_framework使用
    文档地址:https://darkcooking.gitbooks.io/django-rest-framework-cn/content/

    现在我主要教大家如何去实战,做一个简易的知乎日报API
    首先你要熟悉django的基本用法会写模型会写视图函数会配置url

    1.配置字符编码

    因为我们等一下要使用中文,所以要先设好字符编码
    在settings.py里将LANGUAGE_CODE设为'zh-CN'
    然后添加这两行

    FILE_CHARSET='utf-8'
    DEFAULT_CHARSET='utf-8'
    

    还要进入到数据库
    依次输入

    set character_set_client=utf8 ;
    set character_set_connection=utf8 ;
    set character_set_database=utf8 ;
    set character_set_results=utf8 ;
    set character_set_server=utf8 ;
    set character_set_system=utf8 ;
    

    设置完成之后,输入

    show variables like 'character%';
    
    Variable_name Value
    character_set_client utf8
    character_set_connection utf8
    character_set_database utf8
    character_set_filesystem binary
    character_set_results utf8
    character_set_server utf8
    character_set_system utf8
    character_sets_dir /usr/share/mysql/charsets/

    +--------------------------+----------------------------+

    对照一下结果是否是这样

    2.然后开始写模型

    class News(models.Model):
        created = models.DateTimeField(auto_now_add=True)
        title = models.CharField(max_length=100)
        image = models.CharField(max_length=100)
        theme_id =models.IntegerField()
        class Meta:
            ordering = ('created',)
        def __unicode__(self):
            return self.title
    
    class NewsDetail(models.Model):
        news = models.ForeignKey(News)
        created = models.DateTimeField(auto_now_add=True)
        content  = models.CharField(max_length=1000)
        image = models.CharField(max_length=99)
        class Meta:
            ordering = ('created',)
        def __unicode__(self):
            return self.news.title
    

    这里不难理解,这里我分两层,第一层是概括,第二层是详细内容,然后以概括的作为外键
    概括主要包含标题,配图地址和主题
    详细内容包括内容和配图地址

    3.将模式序列化

    这是rest_framework非常强大的一点
    在app里新建一个serializers.py,然后创建继承自ModelSerializer的类

    from rest_framework import serializers
    class NewsSerializer(serializers.ModelSerializer):
        class Meta:
            model = News
            fields = ('id', 'title', 'image', 'theme_id')
    
    class NewsDetailSerializer(serializers.ModelSerializer):
        class Meta:
            model = NewsDetail
            fields = ('id', 'image', 'content')
    

    model为模型,fields为你想要查询显示的字段

    4.然后再写视图函数

    rest_framework.renderers中的JSONRenderer可以将对象渲染为json形式的字符串

    from rest_framework.renderers import JSONRenderer
    class JSONResponse(HttpResponse):
        """
        用于返回JSON数据.
        """
    
        def __init__(self, data, **kwargs):
            content = JSONRenderer().render(data)
            kwargs['content_type'] = 'application/json'
            content='{"news":'+content+'}'
            super(JSONResponse, self).__init__(content, **kwargs)
    

    我们在字符串外面在包一个news,这样获取比较方便

    如何像知乎日报的那样获取最新的几个news呢

    
    @csrf_exempt
    def latest_news(request):
        """
        展示最新的10个news.
        """
        if request.method == 'GET':
            news = News.objects.all()[:10]
            serializer = NewsSerializer(news, many=True)
            return JSONResponse(serializer.data)
    

    因为返回的是一个集合所以NewsSerializer的many参数要设为True

    返回某个主题的前几个news

    
    @csrf_exempt
    def theme_news(request,theme_id):
        """
        展示某个主题的前10个news.
        """
        if request.method == 'GET':
            news=News.objects.filter(theme_id=theme_id)[:10]
            serializer = NewsSerializer(news, many=True)
            return JSONResponse(serializer.data)
    

    返回某个id的news的详细内容

    
    @csrf_exempt
    def news_detail(request,news_id):
        """
        显示某个news的内容.
        """
        try:
            news = NewsDetail.objects.get(news_id=news_id)
        except Snippet.DoesNotExist:
            return HttpResponse(status=404)
    
        if request.method == 'GET':
            serializer = NewsDetailSerializer(news)
            # print serializer.data
            return JSONResponse(serializer.data)
    

    这里获取的是单个对象所以不用加many参数

    5.配置url

    urlpatterns = [
        ...
        url(r'^api/4/news/latest$', latest_news),
        url(r'^api/4/news/theme/(?P<theme_id>[0-9]+)/$', theme_news),
        url(r'^api/4/news/(?P<news_id>[0-9]+)/$', news_detail),
    ]
    

    括号包住的是要传入的参数,逗号后面的是匹配url成功后要执行的视图函数
    括号里面有一些正则表达式,自己可以去百度搜一下怎么写

    最后就可以把项目跑起来

    效果如下

    最新消息
    主题消息
    消息详细内容

    相关文章

      网友评论

      • xiaomayi2012:请教博主如何用这个框架 分页 和显示总数啊
        饥渴计科极客杰铿:@xiaomayi2012 自己去学习lavarel数据提取部分
      • 田飞雨:为啥不用 Mixins 和类视图
        田飞雨:@饥渴计科极客杰铿 知乎日报的那些连接怎么弄
        田飞雨:@饥渴计科极客杰铿 看你那个教程中有好多方法,我想问一般都用哪种方法,使用Routers好像很简单
        饥渴计科极客杰铿:@田飞雨 其实都可以,但也有些不同,类视图的话的确很方便,但感觉这个自由度更高,mixins我用postman模拟时会出现认证问题
      • 代号027:用django来做API太重了
        田飞雨:@一点简单 不会 flask :sob:
        代号027:@田飞雨 推荐flask
        田飞雨:@一点简单 还可以用啥做啊
      • patrick_guan:用api_view代码应该更简洁一些:blush:

      本文标题:Python日记——用Django做一个简易的知乎日报API

      本文链接:https://www.haomeiwen.com/subject/qutujttx.html