SQLAlchemy的类继承、抽象类

栏目: 后端 · 前端 · 发布时间: 5年前

内容简介:Python中的类继承是相当容易的,但是在SQLAlchemy中却不能直接用Python类继承完成,还要多加一些设置。网上关于这个东西,东说西说的非常多,甚至官网都没有把最简单的解决方案po出来,取而代之的是非常复杂的首先说最简单的方案,来自Stackoverflow,亲测完美有效,最符合Python类继承。

Python中的类继承是相当容易的,但是在SQLAlchemy中却不能直接用 Python 类继承完成,还要多加一些设置。

网上关于这个东西,东说西说的非常多,甚至官网都没有把最简单的解决方案po出来,取而代之的是非常复杂的 Inheritance Configuration

首先说最简单的方案,来自Stackoverflow,亲测完美有效,最符合Python类继承。

参考:Sqlalchemy: avoiding multiple inheritance and having abstract base class

正解

在这里,我们称这个方法为 __abstract__ 方法:

Base = declarative_base()

class CommonRoutines(Base):
    __abstract__ = True

    id = Column(Integer, primary_key=True)

    def __init__(self):
        # ...

class Foo(CommonRoutines):
    __tablename__ = 'foo'

    name = Column(...)

    def __init__(self, name):
        super().__init__()
        self.name = name
        # ...

也就是说,抽象类中只要用 __abstract__ = True 代替 __tablename__ 即可完成一切工作,其它一切都和Python内置的类继承一摸一样了。

继承中的类方法和静态方法

SQLAlchemy的ORM继承,在 classmethodstaticmethod 继承是和Python OOP面向对象的继承方案一致的。

也就是说:

@staticmethod
@classmethod

继承中的外键

奇怪的是,SQLAlchemy定义的ORM,在继承父级ORM时候, Foreign Key 外键是不能继承的,它强制要求在子类中重新定义。

参考官方文档:Mapping Class Inheritance Hierarchies 建议直接用 Ctrl-f 搜索"foreign`关键字,就能看到官方在继承时,也都要重新定义一遍外键。

再参考:SQLAlchemy Inheritance

class Parent(Base):
    __abstract__ = True

    id = Column('id', Integer, primary_key=True)
    name = Column('name', String)
    age = Column('age', String)
    fk = Column('fk', Integer, ForeignKey('anotherTable.id'), primary_key=True)

class Son(Parent):
    __tablename__ = 'son'

    fk = Column('fk', Integer, ForeignKey('anotherTable.id'), primary_key=True)

其它继承方案

如果参考别人的方案、官网的方案,会让你晕头转向。

为了避免重复参考别人的东西,这里贴上一些 不是解决方案的解决方案

declarative_base(cls=XX) 方法:

class CommonBase(object):
    @classmethod
    def somecommonaction(cls):
        # body here

Base = declarative_base(cls=CommonBase)

class Table1(Base):
    # __tablename__ & Table1 specific fields here

class Table2(Base):
     # __tablename__ & Table2 specific fields here

这样的缺点是,很难看清继承关系。

官方的 __mapper_args__ 方法:

class Person(Base):
    __tablename__ = 'people'
    id = Column(Integer, primary_key=True)
    discriminator = Column('type', String(50))
    __mapper_args__ = {'polymorphic_on': discriminator}

class Engineer(Person):
    __tablename__ = 'engineers'
    __mapper_args__ = {'polymorphic_identity': 'engineer'}
    id = Column(Integer, ForeignKey('people.id'), primary_key=True)
    primary_language = Column(String(50))

可以看出,这个必须在父子类都中分别定义难懂的 __mapper_args__ 属性。这还不算完,官网中还说各种映射需要不同的复杂设置。有兴趣可参考官网: https://docs.sqlalchemy.org/e...


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

六度分隔

六度分隔

邓肯·J·瓦茨 / 陈禹 / 中国人民大学出版社 / 2011-3 / 46.00元

正如副标题所表明的,《六度分隔:一个相互连接的时代的科学》的基本内容是介绍一门正在形成中的新科学——关于网络的一般规律的科学。有这样一门科学吗?它的内容和方法是什么?近年来,这门学科有什么实质性的进展吗?在《六度分隔:一个相互连接的时代的科学》中,作者根据自己的亲身经历娓娓道来,用讲故事的方式,对于这些问题给出了令人信服的回答 除了简要的背景和总结以外,《六度分隔:一个相互连接的时代的科学》......一起来看看 《六度分隔》 这本书的介绍吧!

Markdown 在线编辑器
Markdown 在线编辑器

Markdown 在线编辑器

UNIX 时间戳转换
UNIX 时间戳转换

UNIX 时间戳转换

HEX HSV 转换工具
HEX HSV 转换工具

HEX HSV 互换工具