欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python的Django框架中使用SQLAlchemy操作數(shù)據(jù)庫(kù)的教程

 更新時(shí)間:2016年06月02日 15:53:41   作者:劉天斯  
SQLAlchemy是Python一個(gè)專門的數(shù)據(jù)庫(kù)管理工具,如果對(duì)Django ORM覺(jué)得有些生疏的話完全可以結(jié)合SQLAlchemy,這里我們就來(lái)總結(jié)一下Python的Django框架中使用SQLAlchemy操作數(shù)據(jù)庫(kù)的教程

零、SQLAlchemy是什么?
SQLAlchemy的官網(wǎng)上寫著它的介紹文字:

SQLAlchemy is the Python SQL toolkit and Object Relational Mapper that gives
application developers the full power and flexibility of SQL.
SQLAlchemy 是一個(gè)非常強(qiáng)大的ORM和數(shù)據(jù)庫(kù)工具,但是它龐大的文檔和復(fù)雜的功能總是讓很 多人望而生畏。而Django的ORM相對(duì)來(lái)說(shuō)就讓很多人覺(jué)得簡(jiǎn)單實(shí)用。

事實(shí)上,SQLAlchemy其實(shí)也沒(méi)有那么復(fù)雜,光使用它一些比較高級(jí)的功能其實(shí)并沒(méi)有比 使用Django ORM復(fù)雜多少,而它豐富的功能則能讓你在遇到更復(fù)雜的問(wèn)題時(shí)處理起來(lái)得心應(yīng)手。

寫作本文的主要目的在于:

  • 通過(guò)對(duì)比SQLAlchemy ORM和Django ORM的主要使用方法, 盡量簡(jiǎn)單直觀的讓Django用戶能夠快速了解和上手SQLAlchemy這款強(qiáng)大的工具。
  • 不牽扯到SQLAlchemy具體的技術(shù)細(xì)節(jié),包括Engine連接池、Session的具體工作原理等等

SQLAlchemy相對(duì)于Django內(nèi)建的ORM來(lái)說(shuō),有幾處非常明顯的優(yōu)點(diǎn):

  • 可獨(dú)立使用,任何使用Python的項(xiàng)目都可以用它來(lái)操作數(shù)據(jù)庫(kù)
  • 和直接使用原始的DBAPI相比,提供了非常豐富的特性:連接池、auto-map等等
  • 提供了更底層的SQL抽象語(yǔ)言,能用原始sql解決的問(wèn)題基本上都可以用SQLAlchemy解決
  • 接下來(lái)我們針對(duì)日常的數(shù)據(jù)庫(kù)操作來(lái)對(duì)比一下Django ORM和SQLAlchemy。

文中使用的 SQLAlchemy 版本為 0.9.8

一、Django VS SQLAlchemy

SQLAlchemy的安裝:

 wget http://peak.telecommunity.com/dist/ez_setup.py
 python ez_setup.py
 sudo easy_install sqlalchemy
 sudo easy_install ipython

1.建立數(shù)據(jù)表

首先,我們需要先建立幾個(gè)表。

(1)Django

在Django中,如果要建表,就是在models.py中定義你的數(shù)據(jù)類型:

from django.db import models

class Game(models.Model):
 ... ...

class GameCompany(models.Model):
 ... ...

因?yàn)槲恼轮饕嫦蛴薪?jīng)驗(yàn)的Django用戶,所以此處不寫出詳細(xì)的定義代碼。定義Model以后 我們還需要在settings.py中DATABASES處設(shè)置需要連接的數(shù)據(jù)庫(kù)地址。最后,使用syncdb來(lái) 完成數(shù)據(jù)庫(kù)表的創(chuàng)建。

(2)SQLAlchemy

在SQLAlchemy中,定義表結(jié)構(gòu)的過(guò)程和Django類似:

from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy import Column, Integer, String, ForeignKey, Date
from sqlalchemy.orm import relationship, backref

Base = declarative_base()

# 定義表結(jié)構(gòu)
class GameCompany(Base):
 __tablename__ = 'game_company'

 id = Column(Integer, primary_key=True)
 name = Column(String(200), nullable=False)
 country = Column(String(50))


class Game(Base):
 __tablename__ = 'game'

 id = Column(Integer, primary_key=True)
 company_id = Column(Integer, ForeignKey('game_company.id'), index=True)
 category = Column(String(10))
 name = Column(String(200), nullable=False)
 release_date = Column(Date)

 # 和Django不同,外鍵需要顯式定義,具體好壞見(jiàn)仁見(jiàn)智
 # 此處的relation可以為lazy加載外鍵內(nèi)容時(shí)提供一些可配置的選項(xiàng)
 company = relationship('GameCompany', backref=backref('games'))


# 此處定義要使用的數(shù)據(jù)庫(kù)
engine = create_engine('mysql://root:root@localhost:5379/sqlalchemy_tutorial?charset=utf8')
# 調(diào)用create_all來(lái)創(chuàng)建表結(jié)構(gòu),已經(jīng)存在的表將被忽略
Base.metadata.create_all(engine)

2.插入一些數(shù)據(jù)

接下來(lái),我們往表中插入一些數(shù)據(jù)

(1)Django

Django中比較常用的插入數(shù)據(jù)方法就是使用 .save() 了。

nintendo = GameCompany(name="nintendo", country="Japan")
nintendo.save()

game1 = Game(
 company=nintendo,
 category="ACT",
 name="Super Mario Bros",
 release_date='1985-10-18')
game1.save()

# 或者使用create
Game.objects.create(... ...)

(2)SQLAlchemy

在SQLAlchemy ORM中,有一個(gè)非常關(guān)鍵的對(duì)象 session ,所有對(duì)于數(shù)據(jù)的操作都是 通過(guò)session來(lái)進(jìn)行的,所以要插入數(shù)據(jù)之前,我們得先初始化一個(gè)session:

from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind=engine)
session = Session()

之后插入數(shù)據(jù)的方法也和Django比較相似:

# 添加數(shù)據(jù)
nintendo = GameCompany(name="Nintendo", country="Japan")
capcom = GameCompany(name="Capcom", country="Japan")
game1 = Game(
 company=nintendo,
 category="ACT",
 name="Super Mario Bros",
 release_date='1985-10-18'
)
game2 = Game(
 company=capcom,
 category="ACT",
 name="Devil May Cry 3: Dante's Awakening",
 release_date="2005-03-01",
)
game3 = Game(
 company=nintendo,
 category="RPG",
 name="Mario & Luigi: Dream Team",
 release_date="2013-08-11",
)

# 使用add_all來(lái)讓這些objects和session產(chǎn)生關(guān)系
session.add_all([nintendo, capcom, game1, game2])
# 在沒(méi)有開(kāi)啟autocommit的模式下,不要忘了調(diào)用commit來(lái)讓數(shù)據(jù)寫到數(shù)據(jù)庫(kù)中
session.commit()

除了commit之外,session還有rollback()等方法,你可以把session對(duì)象簡(jiǎn)單看成是一次 transaction,所以當(dāng)你對(duì)內(nèi)容進(jìn)行修改時(shí),需要調(diào)用 session.commit() 來(lái)提交這些修改。

去文檔可以了解更多session相關(guān)內(nèi)容:http://docs.sqlalchemy.org/en/rel_0_9/orm/session.html

二、常用操作

1.簡(jiǎn)單查詢

(1)批量查詢

# -- Django --
Game.objects.filter(category="RPG")

# -- SQLAlchemy --
# 使用filter_by是和django ORM比較接近的方式
session.query(Game).filter_by(category="RPG")
session.query(Game).filter(Game.category == "RPG")

(2)查詢單個(gè)對(duì)象

# -- Django --
Game.objects.get(name="Super Mario Bros")

# -- SQLAlchemy --
session.query(Game).filter_by(name="Super Mario Bros").one()
# `get_objects_or_None()`
session.query(Game).filter_by(name="Super Mario Bros").scalar()

Django中得各種 > 、< 都是使用在字段名稱后面追加 "__gt"、"__lt" 來(lái)實(shí)現(xiàn)的,在SQLAlchemy 中這樣的查詢還要更直觀一些

# -- Django --
Game.objects.filter(release_date__gte='1999-01-01')
# 取反
Game.objects.exclude(release_date__gte='1999-01-01')

# -- SQLAlchemy --
session.query(Game).filter(Game.release_date >= '1999-01-01').count()
# 取反使用 ~ 運(yùn)算符
session.query(Game).filter(~Game.release_date >= '1999-01-01').count()
通過(guò)外鍵組合查詢

# -- Django --
Game.objecs.filter(company__name="Nintendo")

# -- SQLAlchemy --
session.query(Game).join(GameCompany).filter(GameCompany.name == "Nintendo")

2.多條件或查詢

# -- Django --
from django.db.models import Q
Game.objects.filter(Q(category="RPG") | Q(category="ACT"))

# -- SQLAlchemy --
from sqlalchemy import or_
session.query(Game).filter(or_(Game.category == "RPG", Game.category == "ACT"))
session.query(Game).filter((Game.category == "RPG") | (Game.category == "ACT"))

(1)in查詢

# -- Django --
Game.objects.filter(category__in=["GAL", "ACT"])

# -- SQLAlchemy --
session.query(Game).filter(Game.category.in_(["GAL", "ACT"]))

(2)like查詢

# -- Django --
Game.objects.filter(name__contains="Mario")

# -- SQLAlchemy --
session.query(Game.name.contains('Mario'))

3.統(tǒng)計(jì)個(gè)數(shù)

簡(jiǎn)單統(tǒng)計(jì)總數(shù):

# -- Django --
Game.objects.filter(category="RPG").count()

# -- SQLAlchemy --
session.query(Game).filter_by(category="RPG").count()
分組統(tǒng)計(jì)個(gè)數(shù)

# -- Django --
from django.db.models import Count
Game.objects.values_list('category').annotate(Count('pk')).order_by()

# -- SQLAlchemy --
from sqlalchemy import func
session.query(Game.category, func.count(Game.category)).group_by(Game.category).all()

4.結(jié)果排序

對(duì)查詢結(jié)果進(jìn)行排序:

# -- Django --
Game.objects.all().order_by('release_date')
Game.objects.all().order_by('-release_date')
# 多字段排序
Game.objects.all().order_by('-release_date', 'category')

# -- SQLAlchemy --
session.query(Game).order_by(Game.release_date)
session.query(Game).order_by(Game.release_date.desc())
# 多字段排序
session.query(Game).order_by(Game.release_date.desc(), Game.category)

5.修改數(shù)據(jù)

# -- Django --
game = Game.objects.get(pk=1)
game.name = 'Super Mario Brothers'
game.save()

# -- SQLAlchemy --
game = session.query(Game).get(1)
game.name = 'Super Mario Brothers'
session.commit()

6.批量修改

# -- Django --
Game.objects.filter(category="RPG").update(category="ARPG")

# -- SQLAlchemy --
session.query(Game).filter_by(category="RPG").update({"category": "ARPG"})

7.批量刪除

# -- Django --
Game.objects.filter(category="ARPG").delete()

# -- SQLAlchemy --
session.query(Game).filter_by(category="ARPG").delete()

三、SQLAlchemy其他一些值得關(guān)注的功能
上面簡(jiǎn)單列了一些SQLAlchemy ORM和Django ORM的使用方法對(duì)比,SQLAlchemy同時(shí)還提供了一些 其他非常有用的功能,比如Automap~

假如你有一個(gè)Django項(xiàng)目,通過(guò)ORM創(chuàng)建了一大堆Model。這時(shí)來(lái)了一個(gè)新項(xiàng)目,需要操作 這些表,應(yīng)該怎么辦?拷貝這些Models?使用原始的DB-API加上sql來(lái)操作?

其實(shí)使用SQLAlchemy的Automap可以讓你的工作變得非常的方便,你只要在新項(xiàng)目連接到舊數(shù)據(jù)庫(kù),然后 稍微配置一下Automap,就可以使用SQLAlchemy的ORM操作那些通過(guò)別的系統(tǒng)創(chuàng)建的表了。

就像這樣:

from sqlalchemy.ext.automap import automap_base
from sqlalchemy.orm import Session
from sqlalchemy import create_engine

Base = automap_base()
engine = create_engine("sqlite:///mydatabase.db")
Base.prepare(engine, reflect=True)

# user和address就是表明,通過(guò)這樣的語(yǔ)句就可以把他們分別映射到User和Address類
User = Base.classes.user
Address = Base.classes.address

更多信息可以參考詳細(xì)文檔:http://docs.sqlalchemy.org/en/rel_0_9/orm/extensions/automap.html

附:Django與SQLAlchemy結(jié)合的實(shí)例演示
譬如,以下gumi/db.py代碼,其中g(shù)umi制作Django項(xiàng)目名,項(xiàng)目中使用的唯一的數(shù)據(jù)庫(kù)連接的包裝,作為py調(diào)用。

# -*- coding: utf-8 -*- 
from django.conf import settings 
from django.core import signals 
from django.dispatch import dispatcher 
import sqlalchemy 
from sqlalchemy.orm import scoped_session, sessionmaker 
from sqlalchemy.engine.url import URL 
 
__all__ = ['Session', 'metadata'] 
 
def create_engine(): 
 url = URL(drivername=settings.DATABASE_ENGINE, 
    database=settings.DATABASE_NAME, 
    username=settings.DATABASE_USER, 
    password=settings.DATABASE_PASSWORD, 
    host=settings.DATABASE_HOST, 
    port=settings.DATABASE_PORT or None, 
    query = getattr(settings, 'DATABASE_OPTIONS', {}) 
    ) 
 
 options = getattr(settings, 'SQLALCHEMY_OPTIONS', {}) 
 engine = sqlalchemy.create_engine(url, **options) 
 return engine 
 
def end_request(signal, sender): 
 Session.remove() 
 
dispatcher.connect(receiver=end_request, 
     signal=signals.request_finished) 
 
metadata = sqlalchemy.MetaData() 
 
Session = scoped_session(sessionmaker(autoflush=True, 
          transactional=True, 
          bind=create_engine())) 

模塊代碼

from sqlalchemy.orm import * 
from gumi.db import Session, metadata 
some_table = Table('some_table', metadata, 
       Column('id', Integer, primary_key=True), 
       Column('some_value', String(100), nullable=False, 
       mysql_engine='InnoDB', 
       ) 
class SomeObject(object): 
 pass 
mapper(SomeObject, some_table) 

視圖代碼

import django.newforms as forms 
from gumi.db import Session 
 
class SomeForm(forms.Form): 
  # newform 
  pass 
 
def some_action(req): 
  if req.method != "POST": 
   form = SomeForm() 
  else: 
   form = SomeForm(req.POST) 
   if form.is_valid(): 
     data = form.clean() 
     obj = SomeObject() 
     obj.some_param = data['a'] 
     obj.another_param = data['b'] 
     Session.save(obj) 
     Session.commit() 
     return HttpResponseRedirect('/') 
  return render_to_response('some/template.html') 

相關(guān)文章

  • 通過(guò)Python繪制中國(guó)結(jié)的示例代碼

    通過(guò)Python繪制中國(guó)結(jié)的示例代碼

    再過(guò)不久就要到新年了,所以這篇文章將為大家介紹一下如何通過(guò)Python代碼繪制一個(gè)中國(guó)結(jié),文中的示例代碼講解詳細(xì),感興趣的可以動(dòng)手試一試
    2022-01-01
  • python中sort sorted reverse reversed函數(shù)的區(qū)別說(shuō)明

    python中sort sorted reverse reversed函數(shù)的區(qū)別說(shuō)明

    這篇文章主要介紹了python中sort sorted reverse reversed函數(shù)的區(qū)別說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-05-05
  • Appium自動(dòng)化測(cè)試實(shí)現(xiàn)九宮格解鎖

    Appium自動(dòng)化測(cè)試實(shí)現(xiàn)九宮格解鎖

    本文主要介紹了Appium自動(dòng)化測(cè)試實(shí)現(xiàn)九宮格解鎖,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • python使用openpyxl庫(kù)讀寫Excel表格的方法(增刪改查操作)

    python使用openpyxl庫(kù)讀寫Excel表格的方法(增刪改查操作)

    這篇文章主要介紹了python使用openpyxl庫(kù)讀寫Excel表格的方法(增刪改查操作),本文通過(guò)實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-05-05
  • 如何使用Python優(yōu)雅的合并兩個(gè)字典Dict

    如何使用Python優(yōu)雅的合并兩個(gè)字典Dict

    字典是Python語(yǔ)言中唯一的映射類型,在我們?nèi)粘9ぷ髦薪?jīng)常會(huì)遇到,下面這篇文章主要給大家介紹了關(guān)于如何使用Python優(yōu)雅的合并兩個(gè)字典Dict的相關(guān)資料,需要的朋友可以參考下
    2023-05-05
  • 利用Python實(shí)現(xiàn)自定義連點(diǎn)器

    利用Python實(shí)現(xiàn)自定義連點(diǎn)器

    這篇文章主要介紹了如何利用Python實(shí)現(xiàn)自定義連點(diǎn)器,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-08-08
  • PyTorch一小時(shí)掌握之a(chǎn)utograd機(jī)制篇

    PyTorch一小時(shí)掌握之a(chǎn)utograd機(jī)制篇

    這篇文章主要介紹了PyTorch一小時(shí)掌握之a(chǎn)utograd機(jī)制篇,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-09-09
  • python3中的logging記錄日志實(shí)現(xiàn)過(guò)程及封裝成類的操作

    python3中的logging記錄日志實(shí)現(xiàn)過(guò)程及封裝成類的操作

    這篇文章主要介紹了python3中的logging記錄日志實(shí)現(xiàn)過(guò)程及封裝成類的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-05-05
  • Tensorflow 實(shí)現(xiàn)分批量讀取數(shù)據(jù)

    Tensorflow 實(shí)現(xiàn)分批量讀取數(shù)據(jù)

    今天小編就為大家分享一篇Tensorflow 實(shí)現(xiàn)分批量讀取數(shù)據(jù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-01-01
  • python時(shí)間序列數(shù)據(jù)相減的實(shí)現(xiàn)

    python時(shí)間序列數(shù)據(jù)相減的實(shí)現(xiàn)

    本文主要介紹了python時(shí)間序列數(shù)據(jù)相減的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04

最新評(píng)論