电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

瀏覽:5日期:2022-06-27 18:18:18
一、Sqlite3、SQLAlchemy安裝

Sqlite3是Python3標(biāo)準(zhǔn)庫不需要另外安裝,只需要安裝SQLAlchemy即可。本文sqlalchemy版本為1.2.12

pip install sqlalchemy

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

二、ORM操作

除了第一步創(chuàng)建引擎時(shí)連接URL不一樣,其他操作其他mysql等數(shù)據(jù)庫和sqlite都是差不多的。

2.1 創(chuàng)建數(shù)據(jù)庫連接格式說明

sqlite創(chuàng)建數(shù)據(jù)庫連接就是創(chuàng)建數(shù)據(jù)庫,而其他mysql等應(yīng)該是需要數(shù)據(jù)庫已存在才能創(chuàng)建數(shù)據(jù)庫連接;建立數(shù)據(jù)庫連接本文中有時(shí)會(huì)稱為建立數(shù)據(jù)庫引擎。

2.1.1 sqlite創(chuàng)建數(shù)據(jù)庫連接

以相對路徑形式,在當(dāng)前目錄下創(chuàng)建數(shù)據(jù)庫格式如下:

# sqlite://<nohostname>/<path># where <path> is relative:engine = create_engine(’sqlite:///foo.db’)

以絕對路徑形式創(chuàng)建數(shù)據(jù)庫,格式如下:

#Unix/Mac - 4 initial slashes in totalengine = create_engine(’sqlite:////absolute/path/to/foo.db’)#Windowsengine = create_engine(’sqlite:///C:pathtofoo.db’)#Windows alternative using raw stringengine = create_engine(r’sqlite:///C:pathtofoo.db’)

sqlite可以創(chuàng)建內(nèi)存數(shù)據(jù)庫(其他數(shù)據(jù)庫不可以),格式如下:

# format 1engine = create_engine(’sqlite://’)# format 2engine = create_engine(’sqlite:///:memory:’, echo=True)2.1.2 其他數(shù)據(jù)庫創(chuàng)建數(shù)據(jù)庫連接

PostgreSQL:

# defaultengine = create_engine(’postgresql://scott:tiger@localhost/mydatabase’)# psycopg2engine = create_engine(’postgresql+psycopg2://scott:tiger@localhost/mydatabase’)# pg8000engine = create_engine(’postgresql+pg8000://scott:tiger@localhost/mydatabase’)MySQL:

# defaultengine = create_engine(’mysql://scott:tiger@localhost/foo’)# mysql-pythonengine = create_engine(’mysql+mysqldb://scott:tiger@localhost/foo’)# MySQL-connector-pythonengine = create_engine(’mysql+mysqlconnector://scott:tiger@localhost/foo’)# OurSQLengine = create_engine(’mysql+oursql://scott:tiger@localhost/foo’)Oracle:

engine = create_engine(’oracle://scott:tiger@127.0.0.1:1521/sidname’)engine = create_engine(’oracle+cx_oracle://scott:tiger@tnsname’)MSSQL:

# pyodbcengine = create_engine(’mssql+pyodbc://scott:tiger@mydsn’)# pymssqlengine = create_engine(’mssql+pymssql://scott:tiger@hostname:port/dbname’)2.2 創(chuàng)建數(shù)據(jù)庫連接

我們以在當(dāng)前目錄下創(chuàng)建foo.db為例,后續(xù)各步同使用此數(shù)據(jù)庫。

在create_engine中我們多加了兩樣?xùn)|西,一個(gè)是echo=Ture,一個(gè)是check_same_thread=False。

echo=Ture----echo默認(rèn)為False,表示不打印執(zhí)行的SQL語句等較詳細(xì)的執(zhí)行信息,改為Ture表示讓其打印。

check_same_thread=False----sqlite默認(rèn)建立的對象只能讓建立該對象的線程使用,而sqlalchemy是多線程的所以我們需要指定check_same_thread=False來讓建立的對象任意線程都可使用。否則不時(shí)就會(huì)報(bào)錯(cuò):sqlalchemy.exc.ProgrammingError: (sqlite3.ProgrammingError) SQLite objects created in a thread can only be used in that same thread. The object was created in thread id 35608 and this is thread id 34024. [SQL: ’SELECT users.id AS users_id, users.name AS users_name, users.fullname AS users_fullname, users.password AS users_password nFROM users nWHERE users.name = ?n LIMIT ? OFFSET ?’] [parameters: [{}]] (Background on this error at: http://sqlalche.me/e/f405)

from sqlalchemy import create_engineengine = create_engine(’sqlite:///foo.db?check_same_thread=False’, echo=True)

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

2.3 定義映射

先建立基本映射類,后邊真正的映射類都要繼承它

from sqlalchemy.ext.declarative import declarative_baseBase = declarative_base()

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

然后創(chuàng)建真正的映射類,我們這里以一下User映射類為例,我們設(shè)置它映射到users表。

首先要明確,ORM中一般情況下表是不需要先存在的反而為了類與表對應(yīng)無誤借助通過映射類來創(chuàng)建;當(dāng)然表已戲存在了也無可以,在下一小結(jié)中你可以自己決定如果表存在時(shí)要如何操作是重新創(chuàng)建還是使用已有表,但使用已有表你需要確保和類的變量名與表的各字段名要對得上。

from sqlalchemy import Column, Integer, String# 定義映射類User,其繼承上一步創(chuàng)建的Baseclass User(Base): # 指定本類映射到users表 __tablename__ = ’users’ # 如果有多個(gè)類指向同一張表,那么在后邊的類需要把extend_existing設(shè)為True,表示在已有列基礎(chǔ)上進(jìn)行擴(kuò)展 # 或者換句話說,sqlalchemy允許類是表的字集 # __table_args__ = {’extend_existing’: True} # 如果表在同一個(gè)數(shù)據(jù)庫服務(wù)(datebase)的不同數(shù)據(jù)庫中(schema),可使用schema參數(shù)進(jìn)一步指定數(shù)據(jù)庫 # __table_args__ = {’schema’: ’test_database’} # 各變量名一定要與表的各字段名一樣,因?yàn)橄嗤拿质撬麄冎g的唯一關(guān)聯(lián)關(guān)系 # 從語法上說,各變量類型和表的類型可以不完全一致,如表字段是String(64),但我就定義成String(32) # 但為了避免造成不必要的錯(cuò)誤,變量的類型和其對應(yīng)的表的字段的類型還是要相一致 # sqlalchemy強(qiáng)制要求必須要有主鍵字段不然會(huì)報(bào)錯(cuò),如果要映射一張已存在且沒有主鍵的表,那么可行的做法是將所有字段都設(shè)為primary_key=True # 不要看隨便將一個(gè)非主鍵字段設(shè)為primary_key,然后似乎就沒報(bào)錯(cuò)就能使用了,sqlalchemy在接收到查詢結(jié)果后還會(huì)自己根據(jù)主鍵進(jìn)行一次去重 # 指定id映射到id字段; id字段為整型,為主鍵,自動(dòng)增長(其實(shí)整型主鍵默認(rèn)就自動(dòng)增長) id = Column(Integer, primary_key=True, autoincrement=True) # 指定name映射到name字段; name字段為字符串類形, name = Column(String(20)) fullname = Column(String(32)) password = Column(String(32)) # __repr__方法用于輸出該類的對象被print()時(shí)輸出的字符串,如果不想寫可以不寫 def __repr__(self): return '<User(name=’%s’, fullname=’%s’, password=’%s’)>' % ( self.name, self.fullname, self.password)

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

在上面的定義我__tablename__屬性是寫死的,但有時(shí)我們可能想通過外部給類傳遞表名,此時(shí)可以通過以下變通的方法來實(shí)現(xiàn):

def get_dynamic_table_name_class(table_name): # 定義一個(gè)內(nèi)部類 class TestModel(Base): # 給表名賦值 __tablename__ = table_name __table_args__ = {’extend_existing’: True} username = Column(String(32), primary_key=True) password = Column(String(32)) # 把動(dòng)態(tài)設(shè)置表名的類返回去 return TestModel2.4 創(chuàng)建數(shù)據(jù)表

# 查看映射對應(yīng)的表User.__table__# 創(chuàng)建數(shù)據(jù)表。一方面通過engine來連接數(shù)據(jù)庫,另一方面根據(jù)哪些類繼承了Base來決定創(chuàng)建哪些表# checkfirst=True,表示創(chuàng)建表前先檢查該表是否存在,如同名表已存在則不再創(chuàng)建。其實(shí)默認(rèn)就是TrueBase.metadata.create_all(engine, checkfirst=True)# 上邊的寫法會(huì)在engine對應(yīng)的數(shù)據(jù)庫中創(chuàng)建所有繼承Base的類對應(yīng)的表,但很多時(shí)候很多只是用來則試的或是其他庫的# 此時(shí)可以通過tables參數(shù)指定方式,指示僅創(chuàng)建哪些表# Base.metadata.create_all(engine,tables=[Base.metadata.tables[’users’]],checkfirst=True)# 在項(xiàng)目中由于model經(jīng)常在別的文件定義,沒主動(dòng)加載時(shí)上邊的寫法可能寫導(dǎo)致報(bào)錯(cuò),可使用下邊這種更明確的寫法# User.__table__.create(engine, checkfirst=True)# 另外我們說這一步的作用是創(chuàng)建表,當(dāng)我們已經(jīng)確定表已經(jīng)在數(shù)據(jù)庫中存在時(shí),我完可以跳過這一步# 針對已存放有關(guān)鍵數(shù)據(jù)的表,或大家共用的表,直接不寫這創(chuàng)建代碼更讓人心里踏實(shí)

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

從上邊的討論可以知道,我們可以定義model然后根據(jù)model來創(chuàng)建數(shù)據(jù)表(當(dāng)然也可以不創(chuàng)建),那可不可以反過來根據(jù)已有的表來自動(dòng)生成model代碼呢,答案是可以的,使用sqlacodegen。

sqlacodegen安裝操作如下:

# 如果網(wǎng)絡(luò)通,直接pip安裝pip install sqlacodegen# 如果網(wǎng)絡(luò)不通,先在網(wǎng)絡(luò)通的機(jī)器上使用pip下載sqlacodegen及期依賴包pip download sqlacodegen# 上傳到真正要安裝的機(jī)器后再用pip安裝,依賴包也會(huì)自動(dòng)安裝。版本可能會(huì)變化改成自己具體的包名pip install sqlacodegen-2.1.0-py2.py3-none-any.whl

sqlacodegen生成model操作如下:

# linux應(yīng)該被安裝在/usr/local/bin/sqlacodegen# mysql+pymysql示例# 可使用--tables指定要生成model的表,不指定時(shí)為所有表都生成model# 可使用--outfile指定代碼輸出到的文件,不指定時(shí)輸出到stdout# 注意只有當(dāng)表有主鍵時(shí)sqlacodegen才生成如下的class,不然會(huì)使用舊的生成Table()類實(shí)例的形式# 更多說明可使用-h參看sqlacodegen mysql+pymysql://user:password@localhost/dbname [--tables table_name1,table_name2] [--outfile model.py]

如我的一個(gè)示例操作如下,成功為指定表生成model:

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

2.5 建立會(huì)話

增查改刪(CRUD)操作需要使用session進(jìn)行操作

from sqlalchemy.orm import sessionmaker# engine是2.2中創(chuàng)建的連接Session = sessionmaker(bind=engine)# 創(chuàng)建Session類實(shí)例session = Session()

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

2.6 增(向users表中插入記錄)

# 創(chuàng)建User類實(shí)例ed_user = User(name=’ed’, fullname=’Ed Jones’, password=’edspassword’)# 將該實(shí)例插入到users表session.add(ed_user)# 一次插入多條記錄形式session.add_all( [User(name=’wendy’, fullname=’Wendy Williams’, password=’foobar’), User(name=’mary’, fullname=’Mary Contrary’, password=’xxg527’), User(name=’fred’, fullname=’Fred Flinstone’, password=’blah’)])# 當(dāng)前更改只是在session中,需要使用commit確認(rèn)更改才會(huì)寫入數(shù)據(jù)庫session.commit()

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

2.7 查(查詢users表中的記錄)2.7.1 查實(shí)現(xiàn)

query將轉(zhuǎn)成select xxx from xxx部分,filter/filter_by將轉(zhuǎn)成where部分,limit/order by/group by分別對應(yīng)limit()/order_by()/group_by()方法。這句話非常的重要,理解后你將大量減少sql這么寫那在sqlalchemy該怎么寫的疑惑。

filter_by相當(dāng)于where部分,外另可用filter。他們的區(qū)別是filter_by參數(shù)寫法類似sql形式,filter參數(shù)為python形式。

更多匹配寫法見:https://docs.sqlalchemy.org/en/13/orm/tutorial.html#common-filter-operators

our_user = session.query(User).filter_by(name=’ed’).first()our_user# 比較ed_user與查詢到的our_user是否為同一條記錄ed_user is our_user# 只獲取指定字段# 但要注意如果只獲取部分字段,那么返回的就是元組而不是對象了# session.query(User.name).filter_by(name=’ed’).all()# like查詢# session.query(User).filter(User.name.like('ed%')).all()# 正則查詢# session.query(User).filter(User.name.op('regexp')('^ed')).all()# 統(tǒng)計(jì)數(shù)量# session.query(User).filter(User.name.like('ed%')).count()# 調(diào)用數(shù)據(jù)庫內(nèi)置函數(shù)# 以count()為例,都是直接func.func_name()這種格式,func_name與數(shù)據(jù)庫內(nèi)的寫法保持一致# from sqlalchemy import func# session.query(func.count(User3.name)).one()# 字段名為字符串形式# column_name = 'name'# session.query(User).filter(User3.__table__.columns[column_name].like('ed%')).all()# 獲取執(zhí)行的sql語句# 獲取記錄數(shù)的方法有all()/one()/first()等幾個(gè)方法,如果沒加這些方法,得到的只是一個(gè)將要執(zhí)行的sql對象,并沒真正提交執(zhí)行# from sqlalchemy.dialects import mysql# sql_obj = session.query(User).filter_by(name=’ed’)# sql_command = sql_obj.statement.compile(dialect=mysql.dialect(), compile_kwargs={'literal_binds': True})# sql_result = sql_obj.all()

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

另外要注意該鏈接Common Filter Operators節(jié)中形如equals的query.filter(User.name == ’ed’),在真正使用時(shí)都得改成session.query(User).filter(User.name == ’ed’)形式,不然只后看到報(bào)錯(cuò)“NameError: name ’query’ is not defined”。

2.7.2 參數(shù)傳遞問題

我們上邊的sql直接是our_user = session.query(User).filter_by(name=’ed’).first()形式,但到實(shí)際中時(shí)User部分和name=‘ed’這部分是通過參數(shù)傳過來的,使用參數(shù)傳遞時(shí)就要注意以下兩個(gè)問題。

首先,是參數(shù)不要使用引號括起來。比如如下形式是錯(cuò)誤的(使用引號),將報(bào)錯(cuò)sqlalchemy.exc.OperationalError: (sqlite3.OperationalError) no such column

table_and_column_name = 'User'filter = 'name=’ed’'our_user = session.query(table_and_column_name).filter_by(filter).first()

其次,對于有等號參數(shù)需要變換形式。如下去掉了引號,對table_and_column_name沒問題,但filter = (name=’ed’)這種寫法在python是不允許的

table_and_column_name = User# 下面這條語句不符合語法filter = (name=’ed’)our_user = session.query(table_and_column_name).filter_by(filter).first()

對參數(shù)中帶等號的這種形式,現(xiàn)在能想到的只有使用filter代替filter_by,即將sql語句中的=號轉(zhuǎn)變?yōu)閜ython語句中的==。正確寫法如下:

table_and_column_name = Userfilter = (User.name==’ed’)our_user = session.query(table_and_column_name).filter(filter).first()2.8 改(修改users表中的記錄)

# 要修改需要先將記錄查出來mod_user = session.query(User).filter_by(name=’ed’).first()# 將ed用戶的密碼修改為modify_paswdmod_user.password = ’modify_passwd’# 確認(rèn)修改session.commit()# 但是上邊的操作,先查詢再修改相當(dāng)于執(zhí)行了兩條語句,和我們印象中的update不一致# 可直接使用下邊的寫法,傳給服務(wù)端的就是update語句# session.query(User).filter_by(name=’ed’).update({User.password: ’modify_passwd’})# session.commit()# 以同schema的一張表更新另一張表的寫法# 在跨表的update/delete等函數(shù)中synchronize_session=False一定要有不然報(bào)錯(cuò)# session.query(User).filter_by(User.name=User1.name).update({User.password: User2.password}, synchronize_session=False)# 以一schema的表更新另一schema的表的寫法# 寫法與同一schema的一樣,只是定義model時(shí)需要使用__table_args__ = {’schema’: ’test_database’}等形式指定表對應(yīng)的schema

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

2.9 刪(刪除users表中的記錄)

# 要?jiǎng)h除需要先將記錄查出來del_user = session.query(User).filter_by(name=’ed’).first()# 打印一下,確認(rèn)未刪除前記錄存在del_user# 將ed用戶記錄刪除session.delete(del_user)# 確認(rèn)刪除session.commit()# 遍歷查看,已無ed用戶記錄for user in session.query(User): print(user)# 但上邊的寫法,先查詢再刪除,相當(dāng)于給mysql服務(wù)端發(fā)了兩條語句,和我們印象中的delete語句不一致# 可直接使用下邊的寫法,傳給服務(wù)端的就是delete語句# session.query(User).filter_by(name=’ed’).first().delete()

Python3+SQLAlchemy+Sqlite3實(shí)現(xiàn)ORM教程

2.10 直接執(zhí)行SQL語句

雖然使用框架規(guī)定形式可以在一定程度上解決各數(shù)據(jù)庫的SQL差異,比如獲取前兩條記錄各數(shù)據(jù)庫形式如下。

# mssql/accessselect top 2 * from table_name;# mysqlselect * from table_name limit 2;# oracleselect * from table_name where rownum <= 2;

但框架存消除各數(shù)據(jù)庫SQL差異的同時(shí)會(huì)引入各框架CRUD的差異,而開發(fā)人員往往就有一定的SQL基礎(chǔ),如果一個(gè)框架強(qiáng)制用戶只能使用其規(guī)定的CRUD形式那反而增加用戶的學(xué)習(xí)成本,這個(gè)框架注定不能成為成功的框架。直接地執(zhí)行SQL而不是使用框架設(shè)定的CRUD雖然不是一種被鼓勵(lì)的操作但也不應(yīng)被視為一種見不得人的行為。

# 正常的SQL語句sql = 'select * from users'# sqlalchemy使用execute方法直接執(zhí)行SQLrecords = session.execute(sql)

更多關(guān)于Python3 SQLAlchemy Sqlite3相關(guān)教程請查看下面的相關(guān)鏈接

標(biāo)簽: Python 編程
相關(guān)文章:
主站蜘蛛池模板: 安徽合肥格力空调专卖店_格力中央空调_格力空调总经销公司代理-皖格制冷设备 | 北京宣传片拍摄_产品宣传片拍摄_宣传片制作公司-现像传媒 | 深圳品牌设计公司-LOGO设计公司-VI设计公司-未壳创意 | 高速混合机_锂电混合机_VC高效混合机-无锡鑫海干燥粉体设备有限公司 | 影视模板素材_原创专业影视实拍视频素材-8k像素素材网 | CCC验厂-家用电器|服务器CCC认证咨询-奥测世纪 | 河南卓美创业科技有限公司-河南卓美防雷公司-防雷接地-防雷工程-重庆避雷针-避雷器-防雷检测-避雷带-避雷针-避雷塔、机房防雷、古建筑防雷等-山西防雷公司 | 天长市晶耀仪表有限公司| 北京印刷厂_北京印刷_北京印刷公司_北京印刷厂家_北京东爵盛世印刷有限公司 | 济南ISO9000认证咨询代理公司,ISO9001认证,CMA实验室认证,ISO/TS16949认证,服务体系认证,资产管理体系认证,SC食品生产许可证- 济南创远企业管理咨询有限公司 郑州电线电缆厂家-防火|低压|低烟无卤电缆-河南明星电缆 | 真空包装机-诸城市坤泰食品机械有限公司 | sus630/303cu不锈钢棒,440C/430F/17-4ph不锈钢研磨棒-江苏德镍金属科技有限公司 | 编织人生 - 权威手工编织网站,编织爱好者学习毛衣编织的门户网站,织毛衣就上编织人生网-编织人生 | 湖南长沙商标注册专利申请,长沙公司注册代理记账首选美创! | 乙炔气体报警装置|固定式氯化氢检测仪|河南驰诚电气百科 | 精雕机-火花机-精雕机 cnc-高速精雕机-电火花机-广东鼎拓机械科技有限公司 | 沧州友城管业有限公司-内外涂塑钢管-大口径螺旋钢管-涂塑螺旋管-保温钢管生产厂家 | 雷达液位计_超声波风速风向仪_雨量传感器_辐射传感器-山东风途物联网 | 软装设计-提供软装装饰和软装配饰及软装陈设的软装设计公司 | 泰来华顿液氮罐,美国MVE液氮罐,自增压液氮罐,定制液氮生物容器,进口杜瓦瓶-上海京灿精密机械有限公司 | 宿松新闻网 宿松网|宿松在线|宿松门户|安徽宿松(直管县)|宿松新闻综合网站|宿松官方新闻发布 | 订做不锈钢_不锈钢定做加工厂_不锈钢非标定制-重庆侨峰金属加工厂 | 首页_欧瑞传动官方网站--主营变频器、伺服系统、新能源、软起动器、PLC、HMI | 信阳网站建设专家-信阳时代网联-【信阳网站建设百度推广优质服务提供商】信阳网站建设|信阳网络公司|信阳网络营销推广 | 济南菜鸟驿站广告|青岛快递车车体|社区媒体-抖音|墙体广告-山东揽胜广告传媒有限公司 | 高压无油空压机_无油水润滑空压机_水润滑无油螺杆空压机_无油空压机厂家-科普柯超滤(广东)节能科技有限公司 | 南京蜂窝纸箱_南京木托盘_南京纸托盘-南京博恒包装有限公司 | 济南网站策划设计_自适应网站制作_H5企业网站搭建_济南外贸网站制作公司_锐尚 | 玻璃钢型材_拉挤模具_玻璃钢拉挤设备——滑县康百思 | 珠海网站建设_响应网站建设_珠海建站公司_珠海网站设计与制作_珠海网讯互联 | 翅片管换热器「型号全」_厂家-淄博鑫科环保| 卫生型双针压力表-高温防腐差压表-安徽康泰电气有限公司 | 仓储笼_金属箱租赁_循环包装_铁网箱_蝴蝶笼租赁_酷龙仓储笼租赁 测试治具|过炉治具|过锡炉治具|工装夹具|测试夹具|允睿自动化设备 | 讲师宝经纪-专业培训机构师资供应商_培训机构找讲师、培训师、讲师经纪就上讲师宝经纪 | EPK超声波测厚仪,德国EPK测厚仪维修-上海树信仪器仪表有限公司 | 热风机_工业热风机生产厂家上海冠顶公司提供专业热风机图片价格实惠 | 山东钢格板|栅格板生产厂家供应商-日照森亿钢格板有限公司 | 砖机托板价格|免烧砖托板|空心砖托板厂家_山东宏升砖机托板厂 | 潍坊青州古城旅游景点攻略_青州酒店美食推荐-青州旅游网 | 拉曼光谱仪_便携式|激光|显微共焦拉曼光谱仪-北京卓立汉光仪器有限公司 | 防火板_饰面耐火板价格、厂家_品牌认准格林雅|