고정 소수로 Python의 Decimal 인스턴스로 나타난다.
max_digits 숫자에 허용되는 최대 자릿수이다.
decimal_palces 숫자와 함께 저장될 소수 자릿수이다.
DecimalField(…, max_digits=5, decimal_palces=2) 는 최대 999 소수점 2자리 이하이다.
날짜와 시간을 정의한다.
auto_now 는 객체가 저장될 때마다 매번 자동으로 현재시간이 설정된다.( 최근 수정 등 timestamp로써 유용하다.) Model.save가 호출될 때 자동으로 수정된다.
auto_now_add 객체가 처음 생성될 때 자동으로 현재시간이 설정된다. (생성의 timestamp로 유용) 만약 수정을 원한다면 default=today or default=tiemzone.now 를 사용하면된다.
이 3개의 옵션은 같이 쓸 수 없으며, 만약 auto 옵션을 True로 설정하면 editable=False, blank=True 로 설정된다.
ForeignKey(**options)
다른 모델 참조키이다.
models.ForeignKey(Question, on_delete=models.CASCADE) 에서는 각각의 Choice가 하나의 Question에 관계된다는 것을 알려준다.
파일 시스템에서 특정한 디렉터리에 파일이름으로 제한된다.
path 는 필수 인수로 절대 파일 시스템 경로이다.(/home/images)
match : 선택적 인수로 파일 이름을 필터링할 때 사용할 문자열로 된 정규표현식이다. 기본파일 이름에 적용된다. (foo.*\.txt$)
recursive : 선택적 인수로 path의 모든 서브 디렉터리가 포함되야하는지 여부를 지정(true/false)
allow_files or allow_folders : 선택적 인수로 지정된 위치에 파일 or 폴더를 포함할지 여부를 지정한다(true/false). 둘 중 한개는 반드시 True여야한다.
모델 관계
One-to-One
한 테이블의 하나의 레코드가 다른 테이블의 단 하나의 레코드만을 참조할 때, 이 두 모델간의 관계를 일대일 관계라고 한다.
위와 같이 개인 고객 한명당 여권은 한개 있을 수 있다.
OneToOneField
OneToOneField는 ForeignKey 필드에 unique=Ture 옵션을 준 것과 동일하게 동작한다. 즉, ForeignKye 값이 반드시 고유한 값이어야한다.
class 모델명(modesl.Model): 필드명 = models.OneToOneField(관계대상모델)
하나의 테이블의 하나 이상의 레코드가 다른 테이블의 하나 이상의 레코드를 참조하는 경우이다. 다대다 관계를 표현할 때, 두 테이블 사이의 관계를 표현하기 위해 참조 정보를 담은 새로운 테이블(중개모델)을 생성하게 된다.
class 모델명(models.Model): 필드명 = models.ManyToManyField(연결대상모델)
ManyToManyField를이용해서 설정할 수 있다.
classTopping(models.Model): name = models.CharField(max_length=10)classPizza(models.Model): name = models.CharField(max_length=10) toppings = models.ManyToManyField(Topping)
ManyToManyField
다대다 관계의 필드명은 복수형으로 설정하는 것을 권장한다.
서로 관계된 모델들 중 어느 곳에 ModelToManyField를 선언하든 상관 없지만 한 모델에만 선언해야하며, 의미가 자연스로운 곳에 선언해주는 것을 권장한다.
중개모델(Intermediary Model)
두 테이블의 다대다 관계를 나타내주는 모델이다. 두 모델의 외래키를 필드로 가지고 있다. ManyToManyField 로 관계설정하면 자동으로 생성되지만 직접 생성할 수도 있다.
- though
class 모델명(models.Model): 필드명 = models.ManyToManyField(연결대상모델, through='중개모델')
classArtist(models.Model): name = models.CharField(max_length=50)classBand(models.Model): name = models.CharField(max_length=50) members = models.ManyToManyField(Artist, through='Membership')classMembership(models.Model): artist = models.ForignKey(Artist, on_delete=models.CASCADE) band = models.ForignKey(Band, on_delete=models.CASCADE) is_founding_member = models.BooleanField()
- through_field
중개 모델을 직접 생성하는 경우에 두 테이블을 참조하는 외래키 필드를 명확히 선언해야한다.
소스모델(Source Model) : ManyToManyField가 있는 모델을 말한다.
타겟모델(Target Model) : ManyToManyField에 인자로 전달되는 모델을 말한다.
만약 중개 모델에서 하나 이상의 외래키 필드가 소스 모델 혹은 타겟 모델을 참조한다면, through_field 옵션을 통해 외래키 설정을 해줘야한다.
class 타겟모델(models.Model): 필드1 필드2class 소스모델(models.Model): 필드명 = models.ManyToManyField( 타겟모델, through=중개모델, through_fields=('소스필드','타겟필드',) # 반드시 소스필드 타겟필드 순으로 전달해야한다. ) 필드2class 중개모델(models.Model): 타겟필드 = models.ForeignKey(타겟모델) 소스필드 = models.ForeignKey(소스모델) 추가외래키필드 = models.ForeignKey(관계대상모델)
classArtist(models.Model): name = models.CharField(max_length=50)classBand(models.Model): name = models.CharField(max_length=50) members = models.ManyToManyField( Artist, through='Membership', through_fields=('band', 'artist',) )classMembership(models.Model): artist = models.ForignKey(Artist, on_delete=models.CASCADE) band = models.ForignKey(Band, on_delete=models.CASCADE) inviter = models.ForignKey(Artist, on_delete=models.CASCADE)
역참조
외래키 필드를 가진 소스모델에 연결된 타겟모델의 인스턴스들은 자신과 연결된 소스모델의 인스턴스를 가져올 수 있는 Manager 를 가지게된다. Manager는 소스모델명_set 의 형태로 생성된다. Reverse accessor는 관계를 역참조할 수 있는 이 Manger를 의미한다.
생성된 migration들을 실행시켜주고, 자동으로 데이터베이스 스키마를 관리해주는 migrate 명령어가 있다.
sqlmigrate
migration의 이름을 인수로 받아서 실행하는 SQL문을 볼 수 있다. 하지만 실제로 migration을 실행하지는 않는다. 단순히 결과만 출력할 뿐이다.
$pythonmanage.pysqlmigratepolls0001
BEGIN;---- Create model Choice--CREATE TABLE "polls_choice" ("id"integer NOT NULL PRIMARY KEY AUTOINCREMENT, "choice_text"varchar(200) NOT NULL, "votes"integer NOT NULL);---- Create model Question--CREATE TABLE "polls_question" ("id"integer NOT NULL PRIMARY KEY AUTOINCREMENT, "question_text"varchar(200) NOT NULL, "pub_date"datetime NOT NULL);---- Add field question to choice--ALTER TABLE "polls_choice" RENAME TO "polls_choice__old";CREATE TABLE "polls_choice" ("id"integer NOT NULL PRIMARY KEY AUTOINCREMENT, "choice_text"varchar(200) NOT NULL, "votes"integer NOT NULL, "question_id"integer NOT NULL REFERENCES "polls_question" ("id") DEFERRABLE INITIALLY DEFERRED);INSERT INTO "polls_choice" ("id", "choice_text", "votes", "question_id") SELECT "id", "choice_text", "votes", NULL FROM "polls_choice__old";DROP TABLE "polls_choice__old";CREATE INDEX "polls_choice_question_id_c5b4b260" ON "polls_choice" ("question_id");COMMIT;
migrate
아직 적용되지 않은 모든 migration을 수집해 실행한다. django_migrations 테이블에 마이그레이션 적용 여부를 기록한다.
>>> q = Question.objects.get(id=1)>>> q.choice_set.all()<QuerySet []>>>> q.choice_set.create(choice_text='Not much', votes=0)<Choice: Not much>>>> q.choice_set.create(choice_text='The sky', votes=0)<Choice: The sky>>>> q.choice_set.create(choice_text='Just hacking again', votes=0)<Choice: Just hacking again>>>> q.choice_set.all()<QuerySet [<Choice: Not much>,<Choice: The sky>,<Choice: Just hacking again>]>>>> c = Choice.objects.all()>>> c<QuerySet [<Choice: Not much>,<Choice: The sky>,<Choice: Just hacking again>]>