Django ORM解决Oracle表多主键的问题

现状

以Django 3.2为例

Django ORM 设计为默认使用单一主键(通常是自增的 id 字段),这一选择主要基于以下核心原因:

  • 简化ORM设计与操作
    • 统一访问方式
    • 外键关联简化
  • 避免歧义冲突
    • 主键语义明确
    • 防止隐式依赖
  • 性能与数据库兼容
    • 索引效率优化
    • 跨数据库兼容
  • 替代方案成熟
    • unique_together约束

Oracle现状

原始业务表,很多都使用多主键。

使用Django映射现有Oracle数据库无法处理多主键问题。

Oracle表结构 & Django Model

示例(库存表,使用多主键):

|-----------|-------|
| 字段 | 主键 |
| item_no | True |
| warehouse | True |
| location | True |
| quantity | False |

此时,使用Django Inspectdb将该表结构映射到Models文件中,只有item_no字段被设置为primary key,然后使用unique_together的方式来做组合值唯一性控制。这样是存在问题的。

复制代码
class InventoryDetail(models.Model):
    item_no= models.CharField(primary_key=True, max_length=15)
    warehouse = models.CharField(max_length=6)
    location = models.CharField(max_length=6)
    quantity = models.DecimalField(max_digits=18, decimal_places=6, blank=True, null=True)

    class Meta:
        managed = False
        db_table = 'inventory_detail'
        unique_together = (('item_no', 'warehouse', 'location'),)

解决方案

  1. 使用Oracle不可见字段 INVISIBLE,Oracle12以及以上
    1. 为表创建不可见字段id,字段名自定义:

      复制代码
      alter table inventory_detail add (id int invisible  generated as identity)
    2. 不可见字段默认不会出现在select * 结果中,因此不影响现有系统对该表的使用。

    3. 使用了generated as identity,会自动对原始数据的id字段赋值,新插入的数据也会自动赋值。且无法Update该字段的值

  2. Django数据库映射
    1. 使用inspectdb将表结构映射到models文件时,不可见字段无法写入到文件。
    2. 因此需要手动将该字段添加到model中
      1. id = models.BigAutoField(primary_key=True)
      2. unique_together正常情况下会字段写入,可以不用处理
  3. 然后就可以用了

技术细节

Oracle invisible(不可见字段)

Django ORM

小结

目前基本的测试没问题

相关推荐
爱喝可乐的老王1 天前
PyTorch简介与安装
人工智能·pytorch·python
看我干嘛!1 天前
第三次python作业
服务器·数据库·python
deephub1 天前
用 PyTorch 实现 LLM-JEPA:不预测 token,预测嵌入
人工智能·pytorch·python·深度学习·大语言模型
知识分享小能手1 天前
Oracle 19c入门学习教程,从入门到精通,Oracle 的闪回技术 — 语法知识点与使用方法详解(19)
数据库·学习·oracle
好好研究1 天前
Spring Boot - Thymeleaf模板引擎
java·spring boot·后端·thymeleaf
爬山算法1 天前
Hibernate(76)如何在混合持久化环境中使用Hibernate?
java·后端·hibernate
我的xiaodoujiao1 天前
使用 Python 语言 从 0 到 1 搭建完整 Web UI自动化测试学习系列 44--将自动化测试结果自动推送至钉钉工作群聊
前端·python·测试工具·ui·pytest
她说..1 天前
策略模式+工厂模式实现单接口适配多审核节点
java·spring boot·后端·spring·简单工厂模式·策略模式
沈浩(种子思维作者)1 天前
铁的居里点(770度就不被磁铁吸了)道理是什么?能不能精确计算出来?
人工智能·python·flask·量子计算
yufuu981 天前
使用Scikit-learn进行机器学习模型评估
jvm·数据库·python