成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Python與數(shù)據(jù)庫的新人手冊 -- MySQL

liuyix / 3460人閱讀

摘要:首先,使用連接數(shù)據(jù)庫并定義表結(jié)構(gòu)初始化。創(chuàng)建新對象添加到批量添加子非魚虞姬花木蘭提交即保存到數(shù)據(jù)庫結(jié)果查的函數(shù)會返回一個(gè)對象。一對多在的一對多關(guān)系中,使用來表示表的外鍵,表示表與表之間關(guān)聯(lián)的屬性。

Python支持的數(shù)據(jù)庫有很多,MySQL作為主流數(shù)據(jù)庫之一,我們不妨了解下它們之間的小故事
Python操作MySQL的庫有三個(gè),python-MySQL(MySQLdb),PyMySQLSQLAlchemy。

python2中一般使用python-MySQL(MySQLdb),核心由C語言打造,性能最好,缺點(diǎn)是安裝復(fù)雜,已停止更新,不支持python3。

PyMySQL為代替它而生,純python打造,安裝方便,支持python3。

SQLAlchemy是一個(gè)ORM框架,ORM框架的作用就是把數(shù)據(jù)庫表的一行記錄與一個(gè)對象互相做自動轉(zhuǎn)換,它本身無法操作數(shù)據(jù)庫,而是要依賴于MySQLdb、PyMySQL等第三方庫來完成,目前SQLAlchemy在Web編程領(lǐng)域應(yīng)用廣泛。

本文將主要拿SQLAlchemy來進(jìn)行了解學(xué)習(xí)。

安裝工具

首先安裝基本的數(shù)據(jù)庫驅(qū)動pymysql

pip3 install pymysql

然后安裝ORM框架SQLAlchemy

pip3 install sqlalchemy

日常工作中,如果不想每次通過命令行來查看數(shù)據(jù)的話。推薦安裝Navicat for MySQL,通過這個(gè)圖形化工具能夠方便快捷地操作數(shù)據(jù)庫,實(shí)時(shí)查詢數(shù)據(jù)。

初始化數(shù)據(jù)庫

安裝好必要工具后,我們開始嘗試創(chuàng)建個(gè)用戶數(shù)據(jù)user表來。
首先,使用SQLAlchemy連接數(shù)據(jù)庫并定義表結(jié)構(gòu)初始化DBSession。

# 導(dǎo)入SQLAlchemy
from sqlalchemy import Column, BIGINT, String, create_engine
from sqlalchemy.orm import sessionmaker
from sqlalchemy.ext.declarative import declarative_base

# 創(chuàng)建基類
Base = declarative_base()

# 初始化數(shù)據(jù)庫連接:
# "數(shù)據(jù)庫類型+數(shù)據(jù)庫驅(qū)動名稱://用戶名:密碼@數(shù)據(jù)庫地址:端口號/數(shù)據(jù)庫名"
engine = create_engine("mysql+pymysql://root:123123@mysql:3306/test")

# 創(chuàng)建DBSession類型:
DBSession = sessionmaker(bind=engine)
# 創(chuàng)建session對象:
session = DBSession()

# 數(shù)據(jù)庫操作方法
# 初始化數(shù)據(jù)庫
def init_db():
    Base.metadata.create_all(engine)
# 刪除數(shù)據(jù)庫
def drop_db():
    Base.metadata.drop_all(engine)

建立user數(shù)據(jù)表模型:

# 定義user類
class User(Base):
    # 表名
    __tablename__ = "user"

    # 表的結(jié)構(gòu)
    # 設(shè)置id為主鍵 并自增長
    id = Column(BIGINT, primary_key=True, autoincrement=True)
    name = Column(String(20))
    gender = Column(String(2))

# 正式初始化數(shù)據(jù)庫,如果沒有user表的話,這里將自動創(chuàng)建
init_db()

這里有個(gè)需要注意的地方就是在初始化數(shù)據(jù)庫之前需要先定義user數(shù)據(jù)表模型,否則的話無法正常創(chuàng)建user數(shù)據(jù)表。
session(會話),可以看成一個(gè)管理數(shù)據(jù)庫持久連接的對象,后面的操作都將基于session對象進(jìn)行。

如果使用INT自增類型,那么當(dāng)一張表的記錄數(shù)超過2147483647(約21億)時(shí),會達(dá)到上限而出錯(cuò)。使用BIGINT自增類型則可以最多約922億億條記錄。
增刪改查操作

初始化ORM對象后,我們插入一條記錄試試。

# 創(chuàng)建新User對象:
new_user = User(name="mrlizi", gender="man")
# 添加到session:
session.add(new_user)
# 批量添加
session.add_all([
    User(name="子非魚", gender="M"),
    User(name="虞姬", gender="F"),
    User(name="花木蘭", gender="F")
])
# 提交即保存到數(shù)據(jù)庫:
session.commit()

結(jié)果:

Sessionquery函數(shù)會返回一個(gè)Query對象。query函數(shù)可以接受多種參數(shù)類型。

# query: 輸出所有的用戶名
result = session.query(User.name)
# order: 按倒序輸出所有用戶
result = session.query(User).order_by(User.id.desc())
result = session.query(User).order_by(-User.id)
# label: 自定義字段名,查詢結(jié)果可通過item.my_name來獲取用戶名
for item in session.query(User.name.label("my_name")).all()
# filter和filter_by: 篩選用戶名為"mrlizi"的用戶
result = session.query(User).filter(User.name=="mrlizi").one()
result = session.query(User).filter_by(name="mrlizi").one()
# offset和limit:組合起來可做分頁查詢(通過python的切片其實(shí)也一樣),下面的兩種語句的查詢結(jié)果是相同的
result = session.query(User).offset(2).limit(1).all()
result = session.query(User)[1:3]

# AND: 與查詢      
result = session.query(User).filter(and_(User.name=="mrlizi", User.gender=="M")).all()
result = session.query(User).filter(User.name=="mrlizi", User.gender=="M")
result = session.query(User).filter(User.name=="mrlizi").filter(User.gender=="M").all()
# OR: 或查詢
result = session.query(User).filter(or_(User.name == "子非魚", User.name == "花木蘭"))
# 模糊查詢
result = session.query(User).filter(User.name.like("子%")).all()

基本日常用到的查詢方法就是這些,面向?qū)ο蟛僮鞯挠梅ǘ急容^靈活多變,大家可以根據(jù)不同的場景自由組合。

相比去查詢來講,修改就顯得簡單很多,找到命中的記錄,然后通過update方法來進(jìn)行修改。
update方法的synchronize_session參數(shù)用于在更新數(shù)據(jù)后是否對當(dāng)前的session進(jìn)行更新,
synchronize_session = False 不同步更新當(dāng)前session
synchronize_session = "fetch" 更新之前從數(shù)據(jù)庫中拉取實(shí)時(shí)數(shù)據(jù),更新到session對象
synchronize_session = "evaluate" 更新之前先記錄符合的對象,更新完后對記錄的對象進(jìn)行刪除。(意思是不與數(shù)據(jù)庫進(jìn)行同步更新,僅更新當(dāng)前的session記錄)

# 方法一
session.query(User).filter(User.name == "mrlizi").update({"name": "李白"})
# 方法二
user = session.query(User).filter(User.name == "李白").first()
user.name = "鎧"
# 操作方式
result = session.query(User).filter(User.name == "虞姬").update({User.name: "孫尚香"}, synchronize_session="fetch")
# 提交
session.commit()

刪除的話,無非就是查詢到目標(biāo)記錄,然后進(jìn)行刪除。

# 使用查詢語句,filter是where條件,最后調(diào)用delete()進(jìn)行刪除記錄:
session.query(User).filter_by(name="鎧").delete()
session.commit()
關(guān)聯(lián)表查詢

MySQL作為關(guān)系型數(shù)據(jù)庫,可以通過設(shè)置外鍵來進(jìn)行多個(gè)表互相關(guān)聯(lián)查詢。相應(yīng)的,SQLAlchemy也提供了對象之間的一對一、一對多、多對多關(guān)聯(lián)功能。

一對多

SQLAlchemy的一對多關(guān)系中,使用ForeignKey()來表示表的外鍵,relationship()表示表與表之間關(guān)聯(lián)的屬性。

def one_to_many():
    # 定義user類
    class User(Base):
        # 表名
        __tablename__ = "user"

        # 表的結(jié)構(gòu)
        # 設(shè)置id為主鍵 并自增長
        id = Column(BIGINT, primary_key=True, autoincrement=True)
        name = Column(String(20))
        # 定義用戶關(guān)注的公眾號屬性,指明兩者的關(guān)系
        account = relationship("Account", back_populates="user")

    class Account(Base):
        __tablename__ = "account"

        id = Column(BIGINT, primary_key=True, autoincrement=True)
        name = Column(String(20))
        # 設(shè)置外鍵關(guān)聯(lián)到user表的:
        user_id = Column(BIGINT, ForeignKey("user.id"))
        # 定義 Account 的 user 屬性,指明兩者關(guān)系
        user = relationship("User", back_populates="account")

    # 清空數(shù)據(jù)庫并重新初始化
    drop_db()
    init_db()

    mrlizi = User(name="mrlizi")
    mrlizi.account = [
        Account(name="攻城獅峽谷"),
        Account(name="zone7")
    ]
    session.add(mrlizi)

    result = session.query(User).filter(User.name == "mrlizi").one()
    for item in result.account:
        print(item.name)

    result = session.query(Account).filter(Account.name == "攻城獅峽谷").one()
    print(result.user.name)

    session.commit()

one_to_many()

上面代碼的實(shí)現(xiàn)過程:

建立一對多數(shù)據(jù)表模型

將之前的數(shù)據(jù)清空后重新初始化,用新的表模型創(chuàng)建個(gè)新的user,并添加關(guān)注的公眾號account

增加name為mrlizi的user表記錄,同時(shí)創(chuàng)建相關(guān)聯(lián)的公眾號信息記錄。

通過user表查詢相關(guān)聯(lián)的公眾號數(shù)據(jù)

通過account表查詢相關(guān)聯(lián)的用戶數(shù)據(jù)

一對一

一對一其實(shí)就是兩個(gè)表互相關(guān)聯(lián),我們只需要在一對多關(guān)系基礎(chǔ)上的父表中使用uselist參數(shù)來表示。實(shí)現(xiàn)代碼如下:

def one_to_one():
    # 定義user類
    class User(Base):
        __tablename__ = "user"

        id = Column(BIGINT, primary_key=True, autoincrement=True)
        name = Column(String(20))
        account = relationship("Account", uselist=False, back_populates="user")

    # 公眾號類
    class Account(Base):
        __tablename__ = "account"

        id = Column(BIGINT, primary_key=True, autoincrement=True)
        name = Column(String(20))
        # 設(shè)置外鍵關(guān)聯(lián)到user表的:
        user_id = Column(BIGINT, ForeignKey("user.id"))
        # 定義 Account 的 user 屬性,指明兩者關(guān)系
        user = relationship("User", back_populates="account")

    # 清空數(shù)據(jù)庫并重新初始化
    drop_db()
    init_db()

    # 添加記錄
    user = User(name="子非魚")
    user.account = Account(name="攻城獅峽谷")
    session.add(user)
    session.commit()

    # 查詢
    result = session.query(User).filter(User.name == "子非魚").one()
    print(result.account.name)
    # 輸出:
    # 攻城獅峽谷

one_to_one()
多對多

多對多是通過兩個(gè)表之間增加一個(gè)關(guān)聯(lián)的表來實(shí)現(xiàn),這個(gè)關(guān)聯(lián)表使用MetaData對象來與兩個(gè)表關(guān)聯(lián),并用ForeignKey參數(shù)指定鏈接來定位到兩個(gè)不同的表,兩個(gè)不同的表則在relationship()方法中通過secondary參數(shù)來指定關(guān)聯(lián)表。

def many_to_many():
    # 關(guān)聯(lián)表
    association_table = Table("association", Base.metadata,
                              Column("user_id", BIGINT, ForeignKey("user.id")),
                              Column("account_id", BIGINT, ForeignKey("account.id"))
                              )

    class User(Base):
        __tablename__ = "user"

        id = Column(BIGINT, primary_key=True, autoincrement=True)
        name = Column(String(20))
        accounts = relationship("Account", secondary=association_table, back_populates="users")

    class Account(Base):
        __tablename__ = "account"

        id = Column(BIGINT, primary_key=True, autoincrement=True)
        name = Column(String(20))
        users = relationship("User", secondary=association_table, back_populates="accounts")

    # 清空數(shù)據(jù)庫并重新初始化
    drop_db()
    init_db()
    
    # 創(chuàng)建記錄
    user1 = User(name="子非魚")
    user2 = User(name="zone")
    user3 = User(name="mrlizi")
    account1 = Account(name="攻城獅峽谷")
    account2 = Account(name="zone7")

    # 關(guān)聯(lián)記錄
    user1.accounts = [account1]
    user2.accounts = [account1, account2]
    user3.accounts = [account2]

    # 添加并保存
    session.add(user1)
    session.add(user2)
    session.add(user3)
    session.commit()

    # 雙向查詢
    result1 = session.query(User).filter(User.name == "zone").one()
    for item in result1.accounts:
        print(item.name)
    result2 = session.query(Account).filter(Account.name == "攻城獅峽谷").one()
    for item in result2.users:
        print(item.name)

many_to_many()
總結(jié)

MySQL作為主流的數(shù)據(jù)庫之一,我們不一定說要多深入去研究它的使用,但起碼的了解還是要有的。而且python中使用MySQL還是挺簡單的,代碼敲著敲著就會了。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/17900.html

相關(guān)文章

  • Python數(shù)據(jù)庫新人手冊 -- MySQL

    摘要:首先,使用連接數(shù)據(jù)庫并定義表結(jié)構(gòu)初始化。創(chuàng)建新對象添加到批量添加子非魚虞姬花木蘭提交即保存到數(shù)據(jù)庫結(jié)果查的函數(shù)會返回一個(gè)對象。一對多在的一對多關(guān)系中,使用來表示表的外鍵,表示表與表之間關(guān)聯(lián)的屬性。 Python支持的數(shù)據(jù)庫有很多,MySQL作為主流數(shù)據(jù)庫之一,我們不妨了解下它們之間的小故事Python操作MySQL的庫有三個(gè),python-MySQL(MySQLdb),PyMySQL跟...

    endiat 評論0 收藏0
  • Slog3_如何使用PythonMysql進(jìn)行數(shù)據(jù)交互

    摘要:現(xiàn)在,對接文件已經(jīng)到位接下來測試數(shù)據(jù)庫,如果還沒安裝或者安裝了還沒配置,移步數(shù)據(jù)庫配置,完成數(shù)據(jù)庫的安裝和配置或者在官方下載,安裝官方手冊安裝。 ArthurSlog SLog-3 Year·1 Guangzhou·China July 9th 2018 showImg(https://segmentfault.com/img/remote/1460000016093266?w=2...

    ccj659 評論0 收藏0

發(fā)表評論

0條評論

liuyix

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<