Python/DRF

[DRF] DRF Pagination (by. LimitOffsetPagination)

yubi5050 2022. 11. 10. 04:20

Django에서 게시판을 만드는 방법은 크게 아래 두가지 방향으로 구현이 가능하다. 

  • Django의 Paginator (django.core.paginator import Paginator)
  • DRF(Django Rest Framework)의 Pagination(rest_frameowork.pagination)

 

두 방식은 각기 장단점이 있으며, 서비스(기능) 중심으로 더 결이 잘 맞는 적절한 방향을 정하는 것이 좋다. 

 

추가적으로 DRF 에는 세부적으로 3가지 방법이 제공되는데

  • PageNumberPagination : 페이지 크기(PAGE_SIZE)를 정하고, 페이지 번호를 요청하는 조회 방법
  • LimitOffsetPagination : 특정 지점(offset) 부터 특정 데이터 갯수(limit)을 정해서 가져오는 방법
  • CursorPagination : 요청한 데이터 기준 이전, 이후 요청시 사용

 

해당 글은 DRF의 LimitOffsetPagination방식을 이용해 Pagination을 구현하는 방법을 설명한다.

 

1. Project, App 생성

django-admin startproject pagination3
django-admin startapp posts

 

2. settings.py

INSTALLED_APPS = [
    ....,
    'posts',
    'rest_framework',
]

REST_FRAMEWORK = {
    'DEFAULT_PAGINATION_CLASS' : 'rest_framework.pagination.LimitOffsetPagination',
}

 

3. models.py 

from django.db import models

# Create your models here.
class Post(models.Model):
    class Meta:
        db_table = "article"
    title = models.CharField(max_length=256)
    content = models.CharField(max_length=256)

 

4-1. pagination3 / urls.py

# pagination3/urls.py
from django.contrib import admin
from django.urls import path, include

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

 

4-2. posts / urls.py

# posts/urls.py
from django.urls import path
from . import views

urlpatterns = [
    path('page/', views.PostPagination3ViewSet.as_view({'get': 'list'})),
]

 

5. serializers.py

from rest_framework import serializers
from .models import Post

class PostSerializer(serializers.ModelSerializer):
    class Meta:
        model = Post
        fields = "__all__"

 

6. views.py

from rest_framework import viewsets
from rest_framework.pagination import LimitOffsetPagination # Limit Offset 기반 Pagination

from .models import Post
from .serializers import PostSerializer

class Pagination3(LimitOffsetPagination): # settings.py의 LimitOffsetPagination 상속
    # 몇개의 Object를 가져올건지
    default_limit = 3 # Default 2 => 3으로 (사실상 PAGE_SIZE와 같은 역할을 함)

# url : posts/page?limit=4&offset=2
class PostPagination3ViewSet(viewsets.ModelViewSet):
    queryset = Post.objects.all()
    serializer_class = PostSerializer
    pagination_class = Pagination3 #

 

7. admins.py 

from django.contrib import admin
from posts.models import Post

admin.site.register(Post)

 

8. 명령어 실행

$ python manage.py makemigrations
$ python manage.py migrate
$ python manage.py createsuperuser
$ python manage.py runserver

 

 

9. Post 데이터 생성

admin 페이지에 접속하여 post 데이터 생성 (10개 이상 생성 권장)

 

API Test (Postman)

API Request

http://127.0.0.1:8000/posts/page?limit=4&offset=2

 

JSON Response

{
    "count": 15,
    "next": "http://127.0.0.1:8000/posts/page/?limit=4&offset=6",
    "previous": "http://127.0.0.1:8000/posts/page/?limit=4",
    "results": [
        {
            "id": 3,
            "title": "ccc",
            "content": "ccc"
        },
        {
            "id": 4,
            "title": "ddd",
            "content": "ddd"
        },
        {
            "id": 5,
            "title": "eee",
            "content": "eee"
        },
        {
            "id": 6,
            "title": "fff",
            "content": "fff"
        }
    ]
}