Python SQLAlchemy --3
本文為 Python SQLAlchemy ORM 一系列教學文:
刪除
學會如何查詢之後,就能夠進行後續的刪除、更新等操作。
同樣地,以幾個範例做為學習的捷徑。
1 |
user_1 = User('user1', 'username1', 'password_1')
|
上述的範例中,將 id 為 1 的 user 查詢出來後,直接呼叫 delete() 方法進行刪除。
呼叫 delete() 後會回傳刪除成功的資料筆數。
更新
事實上,更新也只需要呼叫 update() 並提供欄位名稱與欄位值的 dictionary 做為參數即可。
1 |
user_1 = User('user1', 'username1', 'password_1')
|
表格關聯(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 。
一對多關聯
1 |
# -*- coding: utf-8 -*- |
範例說明:
首先看到第 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 。
多對一關聯範例如下:
1 |
# -*- coding: utf-8 -*- |
多對多關聯
第 4 種關聯的形式為多對多關聯,先前提到一對一、一對多、多對一的關聯都是屬於直接形式的 A ↔ B 型,然而 SQLAlchemy 的多對多做法是用一個中介的 association table 來多對多映對,就成為一種 A ↔ association table ↔ B 的間接關聯形式。
接下來同樣用一個範例說明:
1 |
# -*- coding: utf-8 -*- |
以下是執行結果:
1 |
A: 1 has relationship with |
如果有將 sessionmaker() 的 echo=True 選項打開的話,就可以發現 SQLAlchemy 將 A 與 B 的對應 id 存至 association 的訊息。
如下所示:
1 |
2013-09-01 11:46:20,862 INFO sqlalchemy.engine.base.Engine INSERT INTO association (table_a_id, table_b_id) VALUES (?, ?) |
範例說明:
第 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 的功能。剩下的功能就得靠各位自行探索囉!
Python SQLAlchemy --3的更多相关文章
- python SQLAlchemy
这里我们记录几个python SQLAlchemy的使用例子: 如何对一个字段进行自增操作 user = session.query(User).with_lockmode('update').get ...
- Python SQLAlchemy --2
本文為 Python SQLAlchemy ORM 一系列教學文: 接下來會更深入地探討查詢的使用. 查詢的基本使用法為 session.query(Mapped Class),其後可加 .group ...
- Python SQLAlchemy --1
本文為 Python SQLAlchemy ORM 一系列教學文: SQLAlchemy 大概是目前 Python 最完整的資料庫操作的套件了,不過最令人垢病的是它的文件真的很難閱讀,如果不搭配個實例 ...
- Python SqlAlchemy使用方法
1.初始化连接 from sqlalchemy import create_engine from sqlalchemy.orm import sessionmaker engine = create ...
- python+SQLAlchemy+爬虫
python+SQLAlchemy+爬虫 前面分享了SQLAlchemy的知识,这次我共享一下学习用python开发爬虫再把爬出来的数据放到用SQLAlchemy的数据库上面的知识,当然我这个是带测试 ...
- Python.SQLAlchemy.0
1. SQLAlchemy and You http://lucumr.pocoo.org/2011/7/19/sqlachemy-and-you/ 2. Overview http://docs.s ...
- Python SQLAlchemy基本操作和常用技巧包含大量实例,非常好python
http://www.makaidong.com/%E8%84%9A%E6%9C%AC%E4%B9%8B%E5%AE%B6/28053.shtml "Python SQLAlchemy基本操 ...
- Python SQLAlchemy入门教程
本文将以Mysql举例,介绍sqlalchemy的基本用法.其中,Python版本为2.7,sqlalchemy版本为1.1.6. 一. 介绍 SQLAlchemy是Python中最有名的ORM工具. ...
- Python—sqlalchemy
SQLAlchemy SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作. #Dialect用于和数据API进行交流,根据配置文 ...
随机推荐
- 用javascript比较快速排序和合并排序的优劣
<script> //用来调用排列方法的类 function arr_sort(arr){ var startTime,endTime; var priv_arr = new Array; ...
- extentreports报告插件与testng集成(二)
之前的一篇文章中,是把extentreports 的报告的初始方法写在driver的初始方法中,写报告的方法在testng的 onTest中,这次将这些方法全都拆出来,写在一个方法类中,这个类重现实现 ...
- 【整理】动态加载Web Services
WebClient client = new WebClient(); String url = "http://localhost/MESAPIWebService/MESAPI.asmx ...
- jeecg 扩展封装查询条件 时间段查询
使用jeecg框架开发的小伙伴们知道,添加查询条件,通常是我们加一个配置(query="true")就可以将该字段设置为查询条件.简单方便.但是这样的配置查询条件仅适用于输入框输入 ...
- Android test---CTS
转载 1.下载最新的CTS download 2.准备工作 3.启动CTS测试 3.1 在控制台进入目录android-cts,目录android-cts下有三个文件夹,其中一个是tools. 3.2 ...
- centos 7 挂载大硬盘
对硬盘sdb进行分区 parted -a optimal /dev/sdb 使用GPT格式 mklabel gpt 建立一个主分区 mkpart primary - 显示分区信息 print 退出 q ...
- log4j输出日志到不同文件
1.先看log4j的配置文件 log4j.properties 没有此文件就在根目录下创建一个: log4j.rootLogger=INFO,R,Client log4j.appender.R=org ...
- ASP.NET生成WORD文档,服务器部署注意事项
网上转的,留查备用,我服务器装的office2007所以修改的是Microsoft Office word97 - 2003 文档这一个. ASP.NET生成WORD文档服务器部署注意事项 1.Asp ...
- MES系统学习
MES系统是当今制造型企业信息化的热点,而统一建模语言UML是面向对象建模的标准语言,在软件工程发挥着重要作用.MES系统如何进行UML建模呢,今天和大家重点讨论一下MES系统的UML建模方法,请看本 ...
- spring quartz 定时器时间格式设置
"0/10 * * * * ?" 10秒执行一次 "0 0 12 * * ?"每天中午十二点触发"0 15 10 ? * *"每天早上10: ...