Django REST Framework DRF后端开发

什么是DRF

DRF是一个Python语言,能快速写出REST风格api的开发框架,并可以自动生成上图所示的管理界面。

优点

  • Python语言简洁,代码量少,有许多middleware方便开发,易于理解

缺点

  • 不适合体量特别大的项目
  • 国内普及度不如java项目,可能需要自己踩坑自己填坑

目录

  1. 环境安装
    1.1 安装新版本Python
    1.2 安装数据库
    1.3 安装DRF
    1.4 安装常用轮子
  2. 项目管理
    2.1 创建项目
    2.2 创建app
    2.3 修改设置
  3. 代码编写
    3.1 创建模型
    3.2 模型迁移至数据库
    3.3 模型序列化
    3.4 接口Views
    3.4 路由Routers
  4. 发布上线
    5.1 测试
    5.2 正式发布
  5. 进阶
    5.1 搜索
    5.2 前端传参过滤
    5.3 前端传参排序
    5.4 viewsets的更多功能
    5.5 鉴权
  6. 可能遇到的问题
    6.1 跨域
    6.2 静态文件无法读取

1.1 安装新版本Python

# 查看python版本
$ python -V

建议安装最新版,看网上有些方法装下来的python版本也不是最新的,可以去官网查看最新版安装链接,下载后解压,安装。

# 下载
$ wget https://www.python.org/ftp/python/3.8.3/Python-3.8.3.tgz
# 解压
$ tar zxvf Python-3.8.3.tgz
$ cd Python-3.8.3/  
# 配置
# ./configure
$ ./configure prefix=/usr/local/python3 --enable-optimizations
# 安装
$ make && make install
# 修改环境变量
$ export PATH=$PATH:/usr/local/python3/bin/
# 查看python版本
$ python3 -V
# pip是python的包管理软件,如果没有的话需安装
$ yum -y install python-pip

1.2 安装数据库

mySQL等主流数据库都是支持的, DRF默认用SQLite作为数据库,这里也用SQLite做演示

# 安装SQLite
$ yum install sqlite-devel

1.3 安装Django

(optional)为了使包配置独立于其他的项目,可以创建一个虚拟环境

$ python3 -m venv env
$ source env/bin/activate

要退出虚拟环境,在任意地方deactivate即可

$ deactivate
# 安装Django
$ pip install django
# 安装DRF
$ pip install djangorestframework

1.4 安装常用轮子

$ pip install requests
$ pip install django-filter
$ pip install django-url-filter

2.1 创建项目

# 创建项目
$ django-admin startproject 项目名

2.2 创建app

DRF上project是用来承载各个app,所谓app可以理解为一个个大的功能块,所以需要再创建一个app

# 创建app
$ python manage.py startapp forum

2.3 修改设置

打开根目录下的settings.py

# 修改INSTALLED_APPS
INSTALLED_APPS = [
    ...
    'rest_framework',
    'forum', # 刚创建的app
    'django_filters', # 筛选用
    'requests', # 网络请求用
]
# 修改时区
TIME_ZONE = 'Asia/Shanghai'
# 添加下面的设置
REST_FRAMEWORK = {
    'DEFAULT_FILTER_BACKENDS': ['django_filters.rest_framework.DjangoFilterBackend'],
    'DEFAULT_PAGINATION_CLASS':  'rest_framework.pagination.PageNumberPagination', # 默认分页类
    'PAGE_SIZE': 10,  # 每页数目
}

3.1 创建模型

在项目文件夹下,修改models.py

# models.py
class Article(models.Model):
    title = models.CharField(max_length=100, default='')
    content = models.TextField(default='')
    created = models.DateTimeField(auto_now_add=True)
    last_modify_date = models.DateTimeField(auto_now=True)
    author = models.TextField()
    dingId = models.TextField(default='lost')
    avatar = models.TextField(default='0')
    comments = models.IntegerField(default=0)
    reports = models.IntegerField(default=0)
    deleted = models.BooleanField(default=False)

    class Meta:
        db_table = 'article'

3.2 模型迁移至数据库

迁移是在代码中写好model后,终端上执行下面的语句,会自动在数据库中同步对应的字段

$ python manage.py makemigration
$ python manage.py migrate
# 将会在SQLite建立名为article的表,并有model上定义的字段

3.3 模型序列化

# serializers.py
from rest_framework import serializers
from forum.models import Article
class ArticleSerializer(serializers.ModelSerializer):

    class Meta:
        model = Article
        fields = '__all__' 
        # fields = ('id', 'title', 'content', 'last_modify_date', 'created')

3.4 接口Views

DRF提供了viewsets,下面的4行代码,就实现了文章列表的增删改查

# views.py
from forum.models import Article
from rest_framework import viewsets
class ArticleViewSet(viewsets.ModelViewSet):
    queryset = Article.objects.all()
    serializer_class = ArticleSerializer

3.4 路由Routers

viewsets 写好后,为了使前端能够访问,需要在项目下url.py中配置供前端访问的路径

# urls.py
from rest_framework.routers import DefaultRouter
from forum import views


forumPrefix = 'api/forum/'
forumRouter = DefaultRouter()
forumRouter.register('article', views.ArticleViewSet)


urlpatterns = [
    path('admin/', admin.site.urls),
    path(forumPrefix, include(forumRouter.urls)),
]

4.1 测试发布

根目录下运行:

$ python manage.py runserver

Validating models...

0 errors found
Development server is running at http://127.0.0.1:8000/
Quit the server with CONTROL-C.

看到http://127.0.0.1:8000/的话就证明没问题了,把这个地址复制到浏览器,就可以看到管理文章开头的管理界面,前端需要调用的话只需拼上前面urls.py中写好的path即可,如http://127.0.0.1:8000/api/forum/article

4.1 正式发布

正式发布时,settings.py文件中Debug是要改为false的,否则系统会把所有的请求缓存到内存上,造成进程占用的内存越来越大,而debug=false后,由于静态文件安全性考虑,DRF管理界面css样式就获取不到了,可以临时先通过 --insecure解决

# settings.py
DEBUG = False
$ nohup python3 manage.py runserver --insecure 0.0.0.0:8000 > out.log 2>&1 &
# 查看输出
$ tail -f out.log

至此,这已经是一个完整的并跑起来的后端服务了,但仅这些肯定不够,DRF能做的事情也还有很多

5.1 搜索

# views
    search_fields = ['id', 'title', 'content']

5.2 前端传参过滤

# views
from url_filter.integrations.drf import DjangoFilterBackend


class ArticleViewSet(viewsets.ModelViewSet):
    queryset = Article.objects.all()
    serializer_class = ArticleSerializer

    filter_backends = [DjangoFilterBackend]
    filter_fields = ['selected', 'deleted', 'reports', 'type']

比如我想筛选用户A点赞数为5到10的文章,前端可以直接把筛选条件拼在url后,十分好用!

.../forum/articles/?author=A&likes__range=5,10

详细的使用说明在:https://django-url-filter.readthedocs.io/en/latest/

5.3 前端传参排序

# views
from rest_framework import filters
class ArticleViewSet(viewsets.ModelViewSet):
    queryset = Article.objects.all()
    serializer_class = ArticleSerializer

    filter_backends = [filters.OrderingFilter, DjangoFilterBackend]
    filter_fields = ['selected', 'deleted', 'reports', 'type']
    ordering_fields = ('id', 'created', 'last_modify_date')

比如要按创建时间倒序排列

.../forum/articles/?ordering=-create

真好用!详细说明在:https://www.django-rest-framework.org/api-guide/filtering/

5.4 用@action为viewsets添加更多功能

class ArticleViewSet(viewsets.ModelViewSet):
    queryset = Article.objects.all()
    serializer_class = ArticleSerializer

    filter_backends = [filters.OrderingFilter, DjangoFilterBackend, filters.SearchFilter]
    ordering_fields = ('id', 'created', 'last_modify_date')
    filter_fields = ['selected', 'deleted', 'reports', 'type']
    search_fields = ['id', 'title', 'content']
    permission_classes = [permissions.IsAuthenticatedOrReadOnly]

    @action(detail=True, method=['GET'])
    # detail=False时,对所有集合产生效果,method默认为GET可省略
    def like(self, request, pk=None):
        article = self.get_object()
        article.likes += 1
        article.save()
        return Response(article.likes)

修改viewset的CURD方法

class UserViewSet(viewsets.ViewSet):
  
    def list(self, request):
        pass

    def create(self, request):
        pass

    def retrieve(self, request, pk=None):
        pass

    def update(self, request, pk=None):
        pass

    def partial_update(self, request, pk=None):
        pass

    def destroy(self, request, pk=None):
        pass

5.5 鉴权 Simple JWT

安装Simple JWT

$ pip install djangorestframework-simplejwt

在setting中添加默认鉴权类

REST_FRAMEWORK = {
    ...
    'DEFAULT_AUTHENTICATION_CLASSES': (
        ...
        'rest_framework_simplejwt.authentication.JWTAuthentication',
    )
    ...
}

在urls.py添加对外接口

from rest_framework_simplejwt.views import (
    TokenObtainPairView,
    TokenRefreshView,
)

urlpatterns = [
    ...
    path('api/token/', TokenObtainPairView.as_view(), name='token_obtain_pair'),
    path('api/token/refresh/', TokenRefreshView.as_view(), name='token_refresh'),
    ...
]

用curl获取token

curl \
  -X POST \
  -H "Content-Type: application/json" \
  -d '{"username": "davidattenborough", "password": "boatymcboatface"}' \
  http://localhost:8000/api/token/

...
{
  "access":"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VyX3BrIjoxLCJ0b2tlbl90eXBlIjoiYWNjZXNzIiwiY29sZF9zdHVmZiI6IuKYgyIsImV4cCI6MTIzNDU2LCJqdGkiOiJmZDJmOWQ1ZTFhN2M0MmU4OTQ5MzVlMzYyYmNhOGJjYSJ9.NHlztMGER7UADHZJlxNG0WSi22a2KaYSfd1S-AuT7lU",
  "refresh":"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VyX3BrIjoxLCJ0b2tlbl90eXBlIjoicmVmcmVzaCIsImNvbGRfc3R1ZmYiOiLimIMiLCJleHAiOjIzNDU2NywianRpIjoiZGUxMmY0ZTY3MDY4NDI3ODg5ZjE1YWMyNzcwZGEwNTEifQ.aEoAYkSJjoWH1boshQAaTkf8G3yn0kapko6HFRt7Rh4"
}

返回的两个token,access默认5分钟过期,过期后用refresh刷新,refresh默认1天过期,可以在settings修改过期时间

# Django project settings.py

from datetime import timedelta

...

SIMPLE_JWT = {
    'ACCESS_TOKEN_LIFETIME': timedelta(minutes=5),
    'REFRESH_TOKEN_LIFETIME': timedelta(days=1)
}

详细使用方法:
https://django-rest-framework-simplejwt.readthedocs.io/en/latest/getting_started.html

6.1 跨域

在app下新建middlewares.py

# middlewares.py
from django.utils.deprecation import MiddlewareMixin


class MiddleWare(MiddlewareMixin):
    def process_response(self, request, response):
        response['Access-Control-Allow-Origin'] = "*"
        if request.method == 'OPTIONS':
            response['Access-Control-Allow-Methods'] = "POST, GET, DELETE, OPTIONS, DELETE"
            response['Access-Control-Max-Age'] = 1728000
            response["Access-Control-Allow-Headers"] = "Content-Type, x-requested-with"
            return response

        return response

修改settings

# settings.py
MIDDLEWARE = [
    ...
    'forum.middlewares.MiddleWare',
]

6.2 静态文件无法读取

前面在正式发布时,添加了--insecure,这样做是不安全的,如过去掉的话会造成管理页面css读取不到的问题。https://docs.djangoproject.com/en/3.0/ref/contrib/staticfiles/#module-django.contrib.staticfiles

我看了半天,最后的处理方法是发布前collectstatic收集静态文件,把整个文件夹丢到服务器上,修改setting里STATIC_URL为服务器上文件夹存放的目录

// 收集静态文件
$ python3 manage.py collectstatic
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,544评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,430评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,764评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,193评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,216评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,182评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,063评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,917评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,329评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,543评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,722评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,425评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,019评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,671评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,825评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,729评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,614评论 2 353