5

我正在寻找一个 SQLAlchemy 唯一的解决方案,用于将从表单提交接收到的 dict 转换为数据库中的一系列行,每个提交的字段一个。这是为了处理跨应用程序差异很大的首选项和设置。但是,它很可能适用于创建类似于数据透视表的功能。我在 ETL 工具中看到过这种类型的事情,但我正在寻找一种直接在 ORM 中执行此操作的方法。我找不到任何关于它的文档,但也许我错过了一些东西。

例子:

从表单提交:{"UniqueId":1, "a":23, "b":"Hello", "c":"World"}

我希望它被转换(在 ORM 中),以便它像这样记录在数据库中:

_______________________________________
|UniqueId| ItemName   | ItemValue     |
---------------------------------------
|  1     |    a       |    23         |
---------------------------------------
|  1     |    b       |    Hello      |
---------------------------------------
|  1     |    c       |    World      |
---------------------------------------

选择后,结果将(在 ORM 中)转换回来自每个单独值的一行数据。

---------------------------------------------------
| UniqueId  |  a     |     b      |       c       |

---------------------------------------------------
|   1       |  23    |   Hello    |   World       |

---------------------------------------------------

我会假设在更新时最好的做法是将删除/创建包装在事务中,以便删除当前记录并插入新记录。

ItemNames 的最终列表将保存在单独的表中。

完全开放给更优雅的解决方案,但如果可能的话,希望远离数据库。

我正在使用带有 SQLAlchemy 的 declarative_base 方法。

提前致谢...

干杯,

保罗

4

1 回答 1

9

这是文档中的一个稍作修改的示例,以使用映射到模型中字典的此类表结构:

from sqlalchemy import *
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm.collections import attribute_mapped_collection
from sqlalchemy.ext.associationproxy import association_proxy
from sqlalchemy.orm import relation, sessionmaker

metadata  = MetaData()
Base = declarative_base(metadata=metadata, name='Base')

class Item(Base):

    __tablename__ = 'Item'
    UniqueId = Column(Integer, ForeignKey('ItemSet.UniqueId'),
                      primary_key=True)
    ItemSet = relation('ItemSet')
    ItemName = Column(String(10), primary_key=True)
    ItemValue = Column(Text) # Use PickleType?

def _create_item(ItemName, ItemValue):
    return Item(ItemName=ItemName, ItemValue=ItemValue)

class ItemSet(Base):

    __tablename__ = 'ItemSet'
    UniqueId = Column(Integer, primary_key=True)
    _items = relation(Item,
                      collection_class=attribute_mapped_collection('ItemName'))
    items = association_proxy('_items', 'ItemValue', creator=_create_item)

engine = create_engine('sqlite://', echo=True)
metadata.create_all(engine)

session = sessionmaker(bind=engine)()
data = {"UniqueId": 1, "a": 23, "b": "Hello", "c": "World"}
s = ItemSet(UniqueId=data.pop("UniqueId"))
s.items = data
session.add(s)
session.commit()
于 2010-01-19T11:35:26.413 回答