PostgreSQL数据库设计揭秘:KitchenOwl如何用智能数据模型支撑你的厨房管理

【免费下载链接】kitchenowl KitchenOwl is a self-hosted grocery list and recipe manager. The backend is made with Flask and the frontend with Flutter. Easily add items to your shopping list before you go shopping. You can also create recipes and add items based on what you want to cook. 【免费下载链接】kitchenowl 项目地址: https://gitcode.com/GitHub_Trending/ki/kitchenowl

KitchenOwl是一款开源的自托管购物清单和食谱管理器,后端采用Flask框架,前端使用Flutter构建。它能帮助用户轻松添加购物清单、创建食谱并根据烹饪需求自动添加食材,让厨房管理变得高效而智能。本文将深入剖析KitchenOwl的PostgreSQL数据库设计,揭示其如何通过精心设计的数据模型支撑起强大的厨房管理功能。

数据模型概览:构建厨房管理的基石

KitchenOwl的数据库设计围绕用户的厨房管理需求展开,核心模型包括家庭(Household)、物品(Item)、食谱(Recipe)等,这些模型通过合理的关系设计,实现了数据的高效组织和交互。

家庭模型(Household):多人协作的核心

家庭模型是KitchenOwl的核心,它允许多个用户共享购物清单和食谱。在backend/app/models/household.py中定义了Household类,包含了家庭名称、语言、功能开关等基本信息,以及与物品、购物清单、食谱等模型的关联关系。

class Household(Model):
    __tablename__ = "household"
    id: Mapped[int] = db.Column(db.Integer, primary_key=True)
    name: Mapped[str] = db.Column(db.String(128), nullable=False)
    language: Mapped[str] = db.Column(db.String())
    planner_feature: Mapped[bool] = db.Column(db.Boolean(), nullable=False, default=True)
    expenses_feature: Mapped[bool] = db.Column(db.Boolean(), nullable=False, default=True)
    # 关联关系
    items: Mapped[List["Item"]] = db.relationship("Item", back_populates="household", cascade="all, delete-orphan")
    shoppinglists: Mapped[List["Shoppinglist"]] = db.relationship("Shoppinglist", back_populates="household", cascade="all, delete-orphan")
    recipes: Mapped[List["Recipe"]] = db.relationship("Recipe", back_populates="household", cascade="all, delete-orphan")

家庭模型通过HouseholdMember类实现了多用户协作,支持设置所有者和管理员权限,确保家庭内数据的安全共享。

物品模型(Item):购物清单的基础

物品模型是购物清单的基础,存储了食材的名称、图标、分类等信息。在backend/app/models/item.py中,Item类定义了物品的基本属性,并通过外键关联到家庭和分类。

class Item(Model, DbModelAuthorizeMixin):
    __tablename__ = "item"
    id: Mapped[int] = db.Column(db.Integer, primary_key=True)
    name: Mapped[str] = db.Column(db.String(128))
    icon: Mapped[str | None] = db.Column(db.String(128), nullable=True)
    category_id: Mapped[int | None] = db.Column(db.Integer, db.ForeignKey("category.id"))
    household_id: Mapped[int] = db.Column(db.Integer, db.ForeignKey("household.id"), nullable=False, index=True)
    # 关联关系
    category: Mapped[Optional["Category"]] = db.relationship("Category")
    shoppinglists: Mapped[List["ShoppinglistItems"]] = db.relationship("ShoppinglistItems", back_populates="item", cascade="all, delete-orphan")

物品模型还引入了orderingsupport字段,用于优化购物清单的显示顺序和物品推荐功能,提升用户体验。

购物清单界面

KitchenOwl的购物清单界面,展示了基于Item模型的食材管理功能,支持分类查看和快速搜索。

食谱与计划:智能厨房的核心功能

食谱模型(Recipe):烹饪知识的结构化存储

食谱模型是KitchenOwl的另一个核心,它不仅存储了食谱的基本信息,还通过关联表实现了与食材、标签的多对多关系。在backend/app/models/recipe.py中,Recipe类定义了食谱的名称、描述、烹饪时间、食材列表等信息。

class Recipe(Model, DbModelAuthorizeMixin):
    __tablename__ = "recipe"
    id: Mapped[int] = db.Column(db.Integer, primary_key=True)
    name: Mapped[str] = db.Column(db.String(128))
    description: Mapped[str] = db.Column(db.String())
    time: Mapped[int] = db.Column(db.Integer)
    yields: Mapped[int] = db.Column(db.Integer)
    household_id: Mapped[int] = db.Column(db.Integer, db.ForeignKey("household.id"), nullable=False, index=True)
    # 关联关系
    items: Mapped[List["RecipeItems"]] = db.relationship("RecipeItems", back_populates="recipe", cascade="all, delete-orphan")
    tags: Mapped[List["RecipeTags"]] = db.relationship("RecipeTags", back_populates="recipe", cascade="all, delete-orphan")
    plans: Mapped[List["Planner"]] = db.relationship("Planner", back_populates="recipe", cascade="all, delete-orphan")

RecipeItemsRecipeTags关联表分别实现了食谱与食材、食谱与标签的多对多关系,使得一个食谱可以包含多种食材,同时可以被多个标签标记,方便分类和搜索。

食谱详情界面

食谱详情界面展示了基于Recipe模型的结构化食谱信息,包括食材列表和烹饪步骤。

膳食计划(Planner):智能的烹饪安排

膳食计划功能允许用户将食谱安排到特定日期,系统会自动将所需食材添加到购物清单。这一功能的实现依赖于Planner模型与RecipeHousehold模型的关联。

class Planner(Model, DbModelAuthorizeMixin):
    __tablename__ = "planner"
    id: Mapped[int] = db.Column(db.Integer, primary_key=True)
    cooking_date: Mapped[datetime] = db.Column(db.DateTime)
    recipe_id: Mapped[int] = db.Column(db.Integer, db.ForeignKey("recipe.id"))
    household_id: Mapped[int] = db.Column(db.Integer, db.ForeignKey("household.id"), nullable=False, index=True)
    # 关联关系
    recipe: Mapped["Recipe"] = db.relationship("Recipe", back_populates="plans")

通过膳食计划,用户可以提前安排一周的烹饪计划,系统会根据计划自动生成购物清单,实现了从食谱到购物的无缝衔接。

膳食计划界面

膳食计划界面展示了如何将食谱安排到特定日期,实现智能的烹饪管理。

数据关系设计:高效关联与查询

KitchenOwl的数据库设计充分利用了PostgreSQL的关系型数据库特性,通过合理的外键设计和关联关系,实现了数据的高效查询和管理。

一对多关系:家庭与物品、食谱

家庭与物品、食谱之间是典型的一对多关系。一个家庭可以拥有多个物品和食谱,而每个物品和食谱只能属于一个家庭。这种关系通过在物品和食谱表中设置household_id外键实现。

多对多关系:食谱与食材、标签

食谱与食材、标签之间是多对多关系。一个食谱可以包含多种食材,一种食材也可以出现在多个食谱中;同样,一个食谱可以有多个标签,一个标签也可以标记多个食谱。这种关系通过RecipeItemsRecipeTags关联表实现。

级联删除:数据完整性保障

在关联关系定义中,KitchenOwl使用了cascade="all, delete-orphan"属性,确保当主表记录被删除时,关联表的相关记录也会被自动删除,保障了数据的完整性。例如,当一个家庭被删除时,其下的所有物品、购物清单和食谱也会被自动删除。

性能优化:提升查询效率

为了提升查询效率,KitchenOwl在数据库设计中采用了多种优化策略:

索引设计

household_id等频繁用于查询条件的字段上创建了索引,如item表中的household_id字段:

household_id: Mapped[int] = db.Column(db.Integer, db.ForeignKey("household.id"), nullable=False, index=True)

索引的使用可以显著提升查询速度,尤其是在处理大量数据时。

延迟加载与预加载

在关联关系中,合理使用了延迟加载(lazy loading)和预加载(eager loading)策略。例如,在Recipe模型的items关联中使用了lazy="selectin",实现了关联数据的高效加载:

items: Mapped[List["RecipeItems"]] = db.relationship("RecipeItems", back_populates="recipe", cascade="all, delete-orphan", lazy="selectin", order_by="RecipeItems._name")

自定义查询方法

模型中定义了多种自定义查询方法,如Item.search_nameRecipe.search_name,这些方法利用PostgreSQL的levenshtein函数实现了模糊搜索,提升了用户搜索体验:

@classmethod
def search_name(cls, name: str, household_id: int) -> list[Self]:
    if "postgresql" in db.engine.name:
        return cls.query.filter(
            cls.household_id == household_id,
            func.levenshtein(func.lower(func.substring(cls.name, 1, len(name))), name[:128].lower()) < 4
        ).order_by(
            func.levenshtein(...),
            cls.support.desc()
        ).limit(item_count).all()

财务管理:追踪家庭开销

除了购物清单和食谱管理,KitchenOwl还提供了家庭开销追踪功能。Expense模型记录了家庭的各项支出,ExpenseCategory模型对支出进行分类,帮助用户更好地管理家庭财务。

开销追踪界面

开销追踪界面展示了基于Expense模型的家庭财务管理功能,支持按类别查看和统计支出。

总结:智能厨房管理的数据库基石

KitchenOwl的PostgreSQL数据库设计充分体现了面向对象的思想,通过合理的模型设计和关系定义,为用户提供了强大而高效的厨房管理功能。从家庭协作到购物清单,从食谱管理到膳食计划,再到开销追踪,每个功能都离不开底层数据模型的支撑。

通过本文的揭秘,我们可以看到一个优秀的开源项目如何通过精心的数据库设计来实现复杂的业务需求。无论是数据模型的设计理念,还是性能优化的实践方法,都值得我们在自己的项目中学习和借鉴。

如果你也想拥有一个智能的厨房管理系统,可以通过以下命令克隆项目进行体验:

git clone https://gitcode.com/GitHub_Trending/ki/kitchenowl

KitchenOwl的数据库设计为我们展示了如何将现实世界的厨房管理需求转化为结构化的数据模型,这不仅是技术的体现,更是对用户需求深刻理解的结果。希望本文能帮助你更好地理解KitchenOwl的内部工作原理,也能为你的数据库设计提供一些启发。

【免费下载链接】kitchenowl KitchenOwl is a self-hosted grocery list and recipe manager. The backend is made with Flask and the frontend with Flutter. Easily add items to your shopping list before you go shopping. You can also create recipes and add items based on what you want to cook. 【免费下载链接】kitchenowl 项目地址: https://gitcode.com/GitHub_Trending/ki/kitchenowl

Logo

开源鸿蒙跨平台开发社区汇聚开发者与厂商,共建“一次开发,多端部署”的开源生态,致力于降低跨端开发门槛,推动万物智联创新。

更多推荐