qhfl-2 ContentType組件

通常商城裏有不少的商品,計優惠券對應着活動類型商品,家電是一類商品,食物是一類商品,優惠券對應着不一樣的商品類別。

from django.db import models

class Appliance(models.Model):
    """
    家用電器表
    id name
  冰箱
  電視
  洗衣機
    """
    name = models.CharField(max_length=64)


class Food(models.Model):
    """
    食物表
    id name
 麪包
 牛奶
    """
    name = models.CharField(max_length=32)


class Fruit(models.Model):
    """
    水果表
    id  name
  蘋果
  香蕉
    """
    name = models.CharField(max_length=32)


class Coupon(models.Model):
    """
    優惠券表
    id  name    appliance_id    food_id     fruit_id
  通用優惠券   null            null        null
  冰箱折扣券   1               null        null
  電視折扣券   2               null        null
  蘋果滿減卷   null            null        1
    我每增長一張表就要多增長一個字段
    """
    name = models.CharField(max_length=32)
    appliance = models.ForeignKey(to="Appliance", null=True, blank=True)
    food = models.ForeignKey(to="Food", null=True, blank=True)
    fruit = models.ForeignKey(to="Fruit", null=True, blank=True)<br># 實際上咱們商品的種類會特別的多,致使咱們這張表外鍵愈來愈多
遇到這種一張表要跟多張表進行外鍵關聯的時候~咱們Django提供了ContentType組件~

需求
需求

 

ContentType組件

ContentType是Django的內置的一個應用,能夠追蹤項目中全部的APP和model的對應關係,並記錄在ContentType表中。django

當咱們的項目作數據遷移後,會有不少django自帶的表,其中就有django_content_type表app

ContentType組件應用:ide

  -- 在model中定義ForeignKey字段,並關聯到ContentType表,一般這個字段命名爲content-type測試

  -- 在model中定義PositiveIntergerField字段, 用來存儲關聯表中的主鍵,一般咱們用object_idui

  -- 在model中定義GenericForeignKey字段,傳入上面兩個字段的名字spa

  --  方便反向查詢能夠定義GenericRelation字段rest

基於ContentType建立表結構

from django.db import models
from django.contrib.contenttypes.models import ContentType
from django.contrib.contenttypes.fields import GenericForeignKey, GenericRelation

# Create your models here.


class Food(models.Model):
    """
    id      title
    1       麪包
    2       牛奶
    """
    title = models.CharField(max_length=32)
    # 不會生成字段 只用於反向查詢
    coupons = GenericRelation(to="Coupon")


class Fruit(models.Model):
    """
    id      title
    1       蘋果
    2       香蕉
    """
    title = models.CharField(max_length=32)


# 若是有40張表
# class Coupon(models.Model):
#     """
#     id      title          food_id    fruit_id
#     1       麪包九五折         1         null
#     2       香蕉滿10元減5元    null       2
#     """
#     title = models.CharField(max_length=32)
#     food = models.ForeignKey(to="Food")
#     fruit = models.ForeignKey(to="Fruit")


# class Coupon(models.Model):
#     """
#     id      title        table_id      object_id
#     1       麪包九五折       1             1
#     2       香蕉滿10元減5元  2             2
#     """
#     title = models.CharField(max_length=32)
#     table = models.ForeignKey(to="Table")
#     object_id = models.IntegerField()
#
#
# class Table(models.Model):
#     """
#     id      app_name       table_name
#     1       demo            food
#     2       demo            fruit
#     """
#     app_name = models.CharField(max_length=32)
#     table_name = models.CharField(max_length=32)
view
class Coupon(models.Model):
    title = models.CharField(max_length=32) # 優惠券的名稱
    # 第一步
    content_type = models.ForeignKey(to=ContentType, on_delete=None)  # 商品表
    # 第二步
    object_id = models.IntegerField() # 商品對象的id
    # 第三步 不會生成字段
    content_object = GenericForeignKey("content_type", "object_id")
    #                            # 綁定外鍵關係,防止商品id不在商品表裏面

這樣不管有多少張優惠券,都只須要外鍵關聯Content_type一張表(裏面存着全部表的關係),而後指明對象的id,就能找到優惠券對應的商品code

 

當一張表跟多張表有外鍵關係時,均可以經過 ContentType 組件來創建對象

 

ContentType 查詢blog

demo/views.py

 

from rest_framework.views import APIView
from rest_framework.response import Response
from .models import Food, Coupon
from django.contrib.contenttypes.models import ContentType

class DemoView(APIView):

    def get(self, request):

        # 給麪包建立一個優惠券
        food_obj = Food.objects.filter(id=1).first()
        # Coupon.objects.create(title="麪包九五折", content_type_id=8, object_id=1)  # 比較麻煩,須要找表id
        # Coupon.objects.create(title="雙十一面包九折促銷", content_object=food_obj)

        # 查詢麪包都有哪些優惠券
        coupons = food_obj.coupons.all()
        print(coupons)
        # 優惠券查對象
        coupon_obj = Coupon.objects.filter(id=1).first()
        content_obj = coupon_obj.content_object
        print(coupon_obj.title)

"""
<QuerySet [<Coupon: Coupon object (1)>, <Coupon: Coupon object (2)>, <Coupon: Coupon object (3)>, <Coupon: Coupon object (4)>]>
麪包九五折
"""

        # 經過ContentType表找表模型
        content = ContentType.objects.filter(app_label="demo", model="food").first()
        print(content)
        model_class = content.model_class()
        ret = model_class.objects.all()
        print(ret)
"""food
<QuerySet [<Food: Food object (1)>, <Food: Food object (2)>]>
        return Response("ContentType測試")
"""

 

 

image

相關文章
相關標籤/搜索