문돌이 존버/Django 스터디
장고(Django), 다양한 응답의 함수 기반 뷰(1)
애뚱
2021. 9. 12. 23:31
반응형
본 글은 Holix의 "리액트와 함께 장고 시작하기 Complete" 강의를 듣고 작성한 일지입니다.
뷰(View)
기본적인 장고 뷰의 로직은 1개의 HTTP 요청에 대해 1개의 뷰가 호출되는 것입니다. 구체적으로 말하면 urls.py/urlpatterns 리스트에 매핑된 호출 가능한 객체를 가리키는데, 웹 클라이언트로부터의 HTTP 요청을 처리합니다. 뷰에는 크게 2가지 형태가 있으며, 1) 함수 기반 뷰(Function Based View, FBV), 2) 클래스 기반 뷰(Class Based View, CBV)입니다.
FBV는 호출 가능한 객체 그 자체이며 장고 뷰의 기본입니다. CBV는 클래스.as_view() 를 통해 호출 가능한 객체를 생성하거나 리턴합니다.
View 호출 시 인자
- HttpRequest
- URL Caputured Values
첫 번째 인자: HttpRequest 객체
- 현재 요청에 대한 모든 내역을 담고 있음
두 번째 인자: 현재 요청의 URL로부터 Capture된 문자열들
- url/re_path를 통한 처리에선 모든 인자가 string 타입으로 전달
- path를 통한 처리에선 매핑된 Converter의 to_python에 맞게 변환된 값이 인자로 전달
from . import views
from django.urls import path, re_path
urlpatterns = [
path('', views.post_list, name='post_list'),
path('<int:pk>/', views.post_detail),
# re_path(r'(?P<pk>\d+/$', views.post_detail) <- 이전 버전
]
위 코드에서 <int:pk> 부분이 Converter이고 pk를 integer로 명시했기 때문에 정수로 변환한 뒤 인자로 전달됩니다.
View 호출에 대한 리턴값
1. HttpResponse 객체를 리턴해야 한다.
- 장고 Middleware에선 뷰에서 HttpResponse 객체를 리턴하기를 기대합니다. 다른 타입을 리턴하면 Middleware에서 처리하는 데 오류가 발생합니다.
(뷰를 미들웨어가 감싸고 있다고 생각하면 됩니다.)
- django.shortcuts.render 함수는 템플릿 응답을 위한 shortcut 함수입니다.
2. 파일과 같은 객체 혹은 str/bytes 타입의 응답을 지원
- str 문자열을 직접 utf8로 인코딩할 필요가 없습니다(장고 디폴트 설정에서 utf8로 인코딩)
- response=HttpResponse()
3. 파일과 같은 객체
response.write(str 또는 bytes 객체)
아래는 위의 내용을 토대로 한 예시 코드입니다.
# views.py
def post_detail(request: HttpRequest, pk: int) -> HttpResponse:
# request.method
# request.META
# request.GET, request.POST, request.FILES, request.body
content = """
<html>...</html>
"""
response = HttpResponse(content)
response.write(content)
response['Custom-Header'] = 'Custom Header Value'
return response
맛보기(?)로 잠깐 CBV를 살펴보면 아래와 같습니다. FBV 기반으로 했을 때와 비교하면 정말 간단하고 매력적으로 보이지만 그 내부가 어떻게 동작하는지 모른다면 이후에 커스터마이징 할 때 더 어려울 수도 있습니다.
# views.py
from django.views.generic import ListView
# 1번 방법(변할 부분이 없다면 깔끔하고 간단함)
post_list = ListView.as_view(model=Post)
# 2번 방법(Class property, 일단 상속받고 재정의하기 용이함)
class PostListView(ListView):
model = Post
# FBV
# def post_list(request):
# qs = Post.objects.all()
# q = request.GET.get('q', '') # q가 없으면 빈 문자열 리턴
# if q:
# qs = qs.filter(content__icontains=q)
# return render(request, 'blog1/post_list.html', {
# 'post_list': qs,
# 'q': q,
# })
728x90
반응형