SQLAlchemy的类继承、抽象类

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

内容简介: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...


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

查看所有标签

猜你喜欢:

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

重启

重启

米奇•乔尔 / 曲强 / 中信出版社 / 2014-6-10 / 45.00元

罗振宇、丹尼尔•平克、赛斯•高汀、丹•艾瑞里、谢家华、阿里安娜•赫芬顿强烈推荐! 美国亚马逊2013年年度商业&投资类图书榜前20名! 互联网时代五大剧变让企业和个人无处可逃 进化,或被扔在旧时代? 全球顶尖的数字预言家独特分享 商业转型与思维转型的实践指南 当个人变为互联世界中的一个节点,如何开启新的工作方式? 如何与顾客建立直接关系?如何进行实用主义营......一起来看看 《重启》 这本书的介绍吧!

SHA 加密
SHA 加密

SHA 加密工具

HEX CMYK 转换工具
HEX CMYK 转换工具

HEX CMYK 互转工具

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

HEX HSV 互换工具