환경: Python 2.7.5 + Django 1.6
Django를 사용하면 다음과 같이 선언적 방식으로 Form을 정의할 수 있습니다.
# -*- coding: utf-8 -*- from django import forms class SimpleForm(forms.Form): field_a = forms.CharField(max_length=100) field_b = forms.CharField(max_length=100)
작성하기는 매우 편하지만 이 양식을 초기화할 때 문제가 발생합니다. 예를 들면 다음과 같습니다.
from polls.forms import SimpleForm
sf = SimpleForm({'field_a': 'value of field_a', 'field_b':'value of field_b'})
그런 다음 Python 셸에서 dir(sf)를 실행하고 인스턴스에 field_a 및 field_b라는 두 가지 속성이 없음을 확인합니다. sf는 sf.field_a처럼 참조할 수 없습니다. 하지만 분명히 {{ form_name.field_name }} 형식으로 템플릿의 양식 필드를 참조할 수 있습니다.
조사 결과 그 구현 메커니즘이 상당히 구불구불하다는 사실을 발견했습니다. 우선, 폼에서 필드를 참조하려면 어떻게 작성해야 할까요? 이렇게 작성해야 합니다: sf['field_a']
왜 이렇게 작성해야 할까요? 위 코드, django.forms.BaseForm의 __getitem__ 메소드:
def __getitem__(self, name): "Returns a BoundField with the given name." try: field = self.fields[name] except KeyError: raise KeyError('Key %r not found in Form' % name) return BoundField(self, field, name)
이는 BaseForm을 dict와 같은 컨테이너로 변환하므로 위 구문을 사용하여 형태.
또 새로운 질문이 떠오릅니다. 왜 템플릿의 양식 필드를 {{ form_name.field_name }} 형식으로 참조할 수 있나요? Django 공식 문서(https://docs.djangoproject.com/en/1.6/topics/templates/#variables)를 참조하세요. Django의 템플릿 엔진이 {{ form_name.field_name }}과 같은 표현식을 발견하면 form_name 객체에 대한 사전 조회를 실행하므로 템플릿 엔진이 {{ sf.field_a }}를 평가할 때 실제로 sf['를 실행합니다. field_a '], 진실이 나왔습니다.
게다가 위의 SimpleForm 유형은 실제로 django.forms.DeclarativeFieldsMetaclass 입니다. 이 메타클래스는 실제로 SimpleForm에서 선언적 구문으로 선언된 모든 필드(상위 클래스의 선언적 필드 포함)를 get_declared_fields 메소드를 통해 dict로 변환하고 해당 dict의 값을 생성할 클래스에 할당한 후 생성합니다. SimpleForm을 기반으로 한 새로운 클래스.