本文為 Python SQLAlchemy ORM 一系列教學文:

刪除

學會如何查詢之後,就能夠進行後續的刪除、更新等操作。

同樣地,以幾個範例做為學習的捷徑。

123456789
user_1 = User('user1', 'username1', 'password_1')user_2 = User('user2', 'username2', 'password_2')session.add(user_1)session.add(user_2)affected_rows = session.query(User).filter_by(id=1).delete()print('Affected rows:', affected_rows)

if session.query(User).filter_by(id=1).count() == 0:    print('id 1 not found')

上述的範例中,將 id 為 1 的 user 查詢出來後,直接呼叫 delete() 方法進行刪除。

呼叫 delete() 後會回傳刪除成功的資料筆數。

更新

事實上,更新也只需要呼叫 update() 並提供欄位名稱與欄位值的 dictionary 做為參數即可。

123456789
user_1 = User('user1', 'username1', 'password_1')user_2 = User('user2', 'username2', 'password_2')session.add(user_1)session.add(user_2)affected_rows = session.query(User).filter_by(id=1).update({'id':3, 'username': 'sqlalchemy'})print('Affected rows:', affected_rows)

for r in session.query(User):    print(r.id, r.username)

表格關聯(Relationship)

SQLAlchemy ORM 最大的特點就是能夠透過 Python 類別間關聯的建立,實作資料庫表格間的關聯,能夠讓程式開發者很方便的取得相關聯的資料。

而關聯的種類有:

  • One to Many
  • Many to one
  • One to one
  • Many to Many

分別代表一筆資料與另一個表格的資料間的關係。

如果有興趣了解的人可以詳閱 Basic Relationship Patterns 。

接下來同樣用一個範例了解 SQLAlchemy ORM 的表格關聯。範例中,除了原先已經定義過的 User 類別之外,還會再多定義一個 Address 類別,兩者間的關係為一對多,代表一個 user 允許有多個 address 。

一對多關聯

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
# -*- coding: utf-8 -*-import hashlibimport sqlalchemy.ormfrom sqlalchemy import create_enginefrom sqlalchemy import Column, Integer, Stringfrom sqlalchemy.ext.declarative import declarative_basefrom sqlalchemy import ForeignKeyfrom sqlalchemy.orm import relationship, backreffrom sqlalchemy.orm import sessionmakerfrom sqlalchemy import descfrom sqlalchemy.orm import aliasedfrom sqlalchemy import func

Base = declarative_base()

class User(Base):

    __tablename__ = 'user'

    id = Column(Integer, primary_key=True)    name = Column(String)    username = Column(String)    password = Column(String)

    def __init__(self, name, username, password):        self.name = name        self.username = username        self.password = hashlib.sha1(password).hexdigest()

    def __repr__(self):        return "User('{}','{}', '{}')".format(            self.name,            self.username,            self.password        )

class Address(Base):

    __tablename__ = 'user_address'

    id = Column(Integer, primary_key=True)    address = Column(String, nullable=False)    user_id = Column(Integer, ForeignKey('user.id'))    user = relationship('User', backref=backref('address', order_by=id))

    def __init__(self, address):        self.address = address

    def __repr__(self):        return "Address('{}')".format(self.address)

if __name__ == '__main__':    engine = create_engine('sqlite:///:memory:', echo=True)    Base.metadata.create_all(engine)

    Session = sessionmaker(bind=engine)    session = Session()

    user_1 = User('user1', 'username1', 'password_1'.encode('utf-8'))    user_2 = User('user2', 'username2', 'password_2'.encode('utf-8'))    user_1.address = [Address('臺灣凱達格蘭大道')]    user_2.address = [Address('美國紐約時代廣場'), Address('美國華盛頓DC')]    session.add(user_1)    session.add(user_2)

    for r in session.query(Address):        print(r.id, r.address, r.user_id)

    for r in session.query(User):        for a in r.address:            print(r.username, 'living at', a.address)

範例說明:

首先看到第 7, 8 行 import 所需的模組( ForeignKey, relationship, backref ) 。

然後在第 38 - 51 行的部份進行 Address 類別的定義,其中第 44 行定義了一個關聯至 user.id 的 Foreign Key,宣告了它是一個指到 user.id 的 Foreign Key。

並且在第 45 行以 relationship() 方法定義類別 Address 與類別 User 間有一個參照的關聯存在。 SQLAlchemy ORM 就會依照 backref('address', order_by=id) 所定義的關聯,將 User 與 Address 之間以 address 屬性關聯起來。

若第 45 行單單只有 relationship('User') 就代表只是單向的從 Address 到 User 之間的關聯存在(Address → User),但由於我們希望查詢 User 時也能夠得知 User 到那些 Address 的資料,因此就得從 User 關聯回 Address (User → Address),形成一種雙向的關係(User ↔ Address) ,在 SQLAlchemy 中,就稱這種雙向的關係為 backref 。因此在第 45 行可以看到 backref('address', order_by=id),其實就是代表若要從 User 關聯回 Address 就得存取 User 的 address 屬性。

backref – indicates the string name of a property to be placed on the
related mapper’s class that will handle this relationship in the other
direction. The other property will be created automatically when the
mappers are configured. Can also be passed as a backref() object to
control the configuration of the new relationship.

第 60, 61 行分別指定了 user_1 , user_2 的地址,然後在第 64, 65 行將這些資料一併新增到資料庫內(包含 Address 的部份, SQLAlchemy 會自動處理)。

第 70 行查詢資料庫內的所有 user 資料,並且透過 relationship('User', backref=backref('address', order_by=id)) 中所指明的 backref('address', order_by=id),以 address 屬性取得與 User 有關聯的 Address 相關資料。 最後,由於 User 與 Address 之間是一對多的關係,因此需要在第 71 行以一個迴圈將各自的 Address 實例一一取出。

一對一關聯

預設的關聯就是一對多,因此我們可以看到前述的範例是以 list 的進行新增,如 user_2.address = [Address('美國紐約時代廣場'), Address('美國華盛頓DC')]

可是有些時候我們會限制資料間只能有一對一的關係存在,此時就需要在 backref() 中加上 uselist=False 做為參數,以表明一對一的關係。

例如前述範例的第 45 行可變更為:

1
backref('address', uselist=False, order_by=id)

那麼在第 62, 63 行就不需再以 list 的形式指定,同時在第 71 行也不需要再多一個迴圈將各別的 Address 取出。 多對一關聯 同樣以 User, Address 兩個類別作為例子,試想當多個使用者住在同一個地址時,就是多對一的情況。

此時,前述範例就得稍作變更,變成將 Address 的 Foreign Key 移除,而改至 User 中,並將關係指向 Address。同時,我們已不再需要從 Address 雙向關聯回 User ,因此在 relationship() 中也不需再指定 backref

多對一關聯範例如下:

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
# -*- coding: utf-8 -*-import hashlibimport sqlalchemy.ormfrom sqlalchemy import create_enginefrom sqlalchemy import Column, Integer, Stringfrom sqlalchemy.ext.declarative import declarative_basefrom sqlalchemy import ForeignKeyfrom sqlalchemy.orm import relationship, backreffrom sqlalchemy.orm import sessionmakerfrom sqlalchemy import descfrom sqlalchemy.orm import aliasedfrom sqlalchemy import func

Base = declarative_base()

class User(Base):

    __tablename__ = 'user'

    id = Column(Integer, primary_key=True)    name = Column(String)    username = Column(String)    password = Column(String)    address_id = Column(Integer, ForeignKey('user_address.id'))    address = relationship('Address')

    def __init__(self, name, username, password):        self.name = name        self.username = username        self.password = hashlib.sha1(password).hexdigest()

    def __repr__(self):        return "User('{}','{}', '{}')".format(            self.name,            self.username,            self.password        )

class Address(Base):

    __tablename__ = 'user_address'

    id = Column(Integer, primary_key=True)    address = Column(String, nullable=False)

    def __init__(self, address):        self.address = address

    def __repr__(self):        return "Address('{}')".format(self.address)

if __name__ == '__main__':    engine = create_engine('sqlite:///:memory:', echo=True)    Base.metadata.create_all(engine)

    Session = sessionmaker(bind=engine)    session = Session()

    user_1 = User('user1', 'username1', 'password_1'.encode('utf-8'))    user_2 = User('user2', 'username2', 'password_2'.encode('utf-8'))    address_1 = Address('臺灣凱達格蘭大道')    user_1.address = address_1    user_2.address = address_1    session.add(user_1)    session.add(user_2)

    for r in session.query(User):        print(r.username, 'living at', r.address.address)        print('Address_id:', r.address_id)

多對多關聯

第 4 種關聯的形式為多對多關聯,先前提到一對一、一對多、多對一的關聯都是屬於直接形式的 A ↔ B 型,然而 SQLAlchemy 的多對多做法是用一個中介的 association table 來多對多映對,就成為一種 A ↔ association table ↔ B 的間接關聯形式。

接下來同樣用一個範例說明:

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
# -*- coding: utf-8 -*-import hashlibimport sqlalchemy.ormfrom sqlalchemy import create_enginefrom sqlalchemy import Column, Integer, Stringfrom sqlalchemy.ext.declarative import declarative_basefrom sqlalchemy import ForeignKeyfrom sqlalchemy.orm import relationship, backreffrom sqlalchemy.orm import sessionmakerfrom sqlalchemy import descfrom sqlalchemy.orm import aliasedfrom sqlalchemy import funcfrom sqlalchemy import Table

Base = declarative_base()

association_table = Table(    'association',    Base.metadata,    Column('table_a_id', Integer, ForeignKey('table_a.id')),    Column('table_b_id', Integer, ForeignKey('table_b.id')))

class A(Base):    __tablename__ = 'table_a'    id = Column(Integer, primary_key=True)    children = relationship('B', secondary=association_table)

class B(Base):    __tablename__ = 'table_b'    id = Column(Integer, primary_key=True)

if __name__ == '__main__':    engine = create_engine('sqlite:///:memory:')    Base.metadata.create_all(engine)

    Session = sessionmaker(bind=engine)    session = Session()

    b_list = [B(), B(), B()]    a_1 = A()    a_2 = A()    a_1.children = b_list    a_2.children = b_list    session.add(a_1)    session.add(a_2)    session.commit()

    for a in session.query(A):        print('A:', a.id, 'has relationship with')        for b in a.children:            print('\tB:', b.id)

以下是執行結果:

12345678
A: 1 has relationship with    B: 1    B: 2    B: 3A: 2 has relationship with    B: 1    B: 2    B: 3

如果有將 sessionmaker()echo=True 選項打開的話,就可以發現 SQLAlchemy 將 A 與 B 的對應 id 存至 association 的訊息。

如下所示:

12
2013-09-01 11:46:20,862 INFO sqlalchemy.engine.base.Engine INSERT INTO association (table_a_id, table_b_id) VALUES (?, ?)2013-09-02 11:46:20,862 INFO sqlalchemy.engine.base.Engine ((1, 1), (1, 2), (1, 3), (2, 1), (2, 2), (2, 3))

範例說明:

第 17 行定義了一個 association table 用來做為多對多的間接映射的資料表格,Table() 會用 Metadata 來關聯到 declarative base class ,所以 ForeignKey() 就可以分別建立連結到 table_a , table_b

第 25 - 33 行定義了 2 個類別 A, B,其中第 28 行以 relationship('B', secondary=association_table) 指明與 B 之間有關聯存在,並且以 secondary=association_table 指明透過 association_table 建立多對多的關係。 第 43 - 50 行則是建立多對多的資料。

最後在第 52 - 55 行將資料庫內的結果列印出來。

值得注意的是在範例中並沒有建立雙向的關係,如果要查詢 B 時也能夠得知與 A 的關聯,就得在 A 類別的 relationship() 加上 backref=backref('children') 指明雙向關係,如此一來 B 就可以透過存取 children 屬性取得相關聯的 A 。

多對多關聯的刪除

在多對多的關聯下,若刪除一筆資料,並不需要手動更新 association table 內的資料,SQLAlchemy 會自動進行更新。 除了多對多的資料關聯是自動 cascade 刪除之外,其他的關聯就得自行在 relationship() 指明使用 cascade 刪除,例如:

1
addresses = relationship('Address', backref='user', cascade='all, delete, delete-orphan')

詳見 Configuring delete/delete-orphan Cascade

結語

至此,已解說完大部份的 SQLAlchemy ORM 的功能。剩下的功能就得靠各位自行探索囉!

转载于:https://www.cnblogs.com/mosson/p/6257181.html

Python SQLAlchemy --3相关推荐

  1. python sqlalchemy操作SQLite

    日期转时间: from sqlalchemy import Column, Integer, String, Float, Date date = Column(Date) data="20 ...

  2. Python SqlAlchemy使用方法

    1.初始化连接 from sqlalchemy import create_engine from sqlalchemy.orm import sessionmaker engine = create ...

  3. Python SQLAlchemy入门教程

    原文:https://www.cnblogs.com/ybjourney/p/11832045.html Python SQLAlchemy入门教程 一.介绍 关于ORM 为什么用sqlalchemy ...

  4. Python sqlalchemy 连接常用的数据库

    python连接sqlServer数据库,Oracle数据库,MongoDB数据库,mysql数据库 python sqlalchemy 简介 学习链接 sqlalchemy是什么? sqlalche ...

  5. Python SQLAlchemy介绍

    Python SQLAlchemy教程 一. 介绍 关于ORM 二. 使用 概念和数据类型 概念 常见数据类型 使用步骤 创建数据库表 1.安装 2. 创建连接 3. 创建数据库表类(模型) 4. 生 ...

  6. Python SQLAlchemy

    一.写在前面 这篇文章主要介绍了Python的SQLAlchemy框架使用入门,SQLAlchemy框架是Python中用来操作数据库的ORM框架之一,学习过程中主要参考网上现有资料,整理成笔记以便后 ...

  7. Python - SQLAlchemy 子查询

    Python - SQLAlchemy 子查询 Max.Bai 2020-08 1. 子查询需要先执行subquery 2. 子查询的列需要通过c来访问  比如: sub_query.c.alarm ...

  8. python sqlalchemy中文手册-基于Python的SQLAlchemy的操作

    安装 在Python使用SQLAlchemy的首要前提是安装相应的模块,当然作为python的优势,可以到python安装目录下的scripts下,同时按住shift+加上鼠标左键,从而在菜单中打开命 ...

  9. Python SQLAlchemy

    SQLAlchemy介绍 SQLAlchemy是一个基于Python的ORM框架.该框架是建立在DB-API之上,使用关系对象映射进行数据库操作. 简而言之就是,将类和对象转换成SQL,然后使用数据A ...

最新文章

  1. SylixOS 系统初探
  2. 新手如何使用有三AI系统性跟读AI领域的论文
  3. 轮播插件swiper.js?
  4. Android之定时器实现的几种方式和removeCallbacks失效问题详解
  5. astr在python_python学习之初识字符串
  6. 计算机系统-电路设计07-上升沿D触发器的内部电路实现/移位寄存器/串行接口/并行接口
  7. 机器学习算法之手撕SVM-线性(理论)
  8. Linux_Qt:-1: error: cannot find xxx/lib: file format not recognized
  9. sqoop导入/导出
  10. 宋宝华——Linux设备驱动开发详解:基于最新的Linux 4.0内核(第一章)
  11. 连接无线上网服务器能查到吗,网络管理历史-怎样才能不让服务器查到自己上网记录我们单位 – 手机爱问...
  12. potplayer去除播放历史收藏夹
  13. wish新店快速出单的方法
  14. 推荐几款适合孩子玩的编程游戏
  15. 快速接入百度地图定位、描点
  16. R语言学习手记 (1)
  17. 川普的退休生活?不,是AI算法的宅舞演绎。
  18. Java曲线之削峰填谷,Sentinel匀速模式(削峰填谷)
  19. 武汉理工大学计算机学院转专业细则,武汉理工大学转专业
  20. jQurey回车登陆

热门文章

  1. algorithm -- 选择排序
  2. apache ab 测试 apr_socket_connect(): 由于目标机器积极拒绝 无法连接
  3. 2011年的MVP礼包
  4. 框架使用SpringBoot + Spring Security Oauth2 +PostMan
  5. 如何通过Windows Server 2008 R2建立NFS存储
  6. 用 JavaScript 的方式理解递归
  7. 关于Unity中UI中的Image节点以及它的Image组件
  8. I/O多路转接之poll,epoll
  9. python 学习资源收集汇总
  10. C# 视频监控系列(11):H264播放器——封装API[HikPlayM4.dll]