SQLAlchemy介绍

  • 一、SQLAlchemy介绍
  • 二、简单使用
  • 1、执行原生sql(不常用)
  • 2、orm使用
  • (1)单表操作
  • (2)一对多的关系,新增数据
  • 三、scoped_session线程安全
  • 四、基本增删查改
  • 查询
  • 删除
  • 修改
  • as别名
  • :占位符,用params传参数
  • 其他查询操作
  • 通配符
  • 限制(用于分页、区间)
  • 排序
  • 分组
  • having筛选
  • 连表(默认使用forinkey关联)
  • UNION组合(了解)
  • 五、多对多关系的建立和操作
  • 1、models.py
  • 2、基本增删查改
  • 六、flask-sqlalchemy的使用
  • 七、项目分文件使用flask-sqlalchemy + flask-migrate


一、SQLAlchemy介绍

1、SQLAlchemy是一个基于python实现的orm框架,该框架建立在DB API之上,使用关系对象映射进行数据库操作,简而言之是:将类和对象转换成SQL,然后使用数据API执行SQL并获取执行结果
2、是第三方的orm框架,可以独立于web项目使用
3、pip3 install sqlalchemy
4、组成部分
	Engine,框架的引擎
    Connection Pooling,数据连接池
    Dialect,选择连接数据库的DB API种类
    Schema/Types,架构和类型
    SQL Expression Language,SQL表达式
   
5、SQLAlchemy不能创建数据库,可以创建表,创建字段
SQLAlchemy本身无法操作数据库,其必须以来pymsql等第三方插件,Dialect用于和数据API进行交流,根据配置文件的不同调用不同的数据库API,从而实现对数据库的操作,如:
    MySQL-Python
        mysql+mysqldb://<user>:<password>@<host>[:<port>]/<dbname>

    pymysql
        mysql+pymysql://<username>:<password>@<host>/<dbname>[?<options>]

    MySQL-Connector
        mysql+mysqlconnector://<user>:<password>@<host>[:<port>]/<dbname>

    cx_Oracle
        oracle+cx_oracle://user:pass@host:port/dbname[?key=value&key=value...]

    更多:http://docs.sqlalchemy.org/en/latest/dialects/index.html

补充:Django中如何反向生成models

python manage.py inspectdb > app/models.py

二、简单使用

能创建表、删除表,但是不能修改表

1、执行原生sql(不常用)

from sqlalchemy import create_engine
import threading

engine = create_engine(
    "mysql+pymysql://root:111@127.0.0.1:3306/bbs_db?charset=utf8",
    max_overflow=0,   # 超过连接池大小外最多创建的连接
    pool_size=2,  	  # 连接池大小
    pool_timeout=30,  # 池中没有线程最多等待的时间,否则报错
    pool_recycle=-1   # 多久之后对线程池中的线程进行一次连接的回收(重置),-1表示不重置
)

def task(arg):
    conn = engine.raw_connection()
    cursor = conn.cursor()
    cursor.execute("select * from blog_tag")
    result = cursor.fetchall()
    print(result)
    cursor.close()
    conn.close()

for i in range(20):
    t = threading.Thread(target=task, args=(i,))
    t.start()

2、orm使用

models.py

# 通过类,创建表

import datetime
from sqlalchemy.orm import sessionmaker, relationship
from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy import Column, Integer, String, Text, ForeignKey, DateTime, UniqueConstraint, Index
Base = declarative_base()

class Users(Base):
    __tablename__ = 'users'  # 数据库表名称
    id = Column(Integer, primary_key=True)  # id 主键
    name = Column(String(32), index=True, nullable=False)  # name列,索引,不可为空、
    age = Column(Integer)
    # email = Column(String(32), unique=True)
    # #datetime.datetime.now不能加括号,加了括号,以后永远是当前时间
    # ctime = Column(DateTime, default=datetime.datetime.now)
    # extra = Column(Text, nullable=True)

    __table_args__ = (
        # UniqueConstraint('id', 'name', name='uix_id_name'), #联合唯一
        # Index('ix_id_name', 'name', 'email'), #索引
    )


# 一对多,一个爱好,可以有多个人喜欢,关联字段写在多的一方,写在Person中
class Hobby(Base):
    __tablename__ = 'hobby'
    id = Column(Integer, primary_key=True)
    caption = Column(String(50), default='篮球')


class Person(Base):
    __tablename__ = 'person'
    nid = Column(Integer, primary_key=True)
    name = Column(String(32), index=True, nullable=True)
    # "hobby.id"中的hobby指的是tablename而不是类名,uselist=False
    # 外键关联
    # hobby_id跟hobby表的id字段关联
    hobby_id = Column(Integer, ForeignKey("hobby.id"))

    # 跟数据库无关,不会新增字段,只用于快速链表操作
    # 类名,backref用于反向查询
    hobby = relationship('Hobby', backref='pers')

# 把表同步到数据库
def init_db():
    """
    根据类创建数据库表
    :return:
    """
    engine = create_engine(
        "mysql+pymysql://root:123@127.0.0.1:3306/aaa?charset=utf8",
        max_overflow=0,  
        pool_size=5,  
        pool_timeout=30,  
        pool_recycle=-1  
    )
    # 把所有被Base管理的表都创建出来
    Base.metadata.create_all(engine)

def drop_db():
    """
    根据类创建数据库表
    :return:
    """
    engine = create_engine(
        "mysql+pymysql://root:123@127.0.0.1:3306/aaa?charset=utf8",
        max_overflow=0,  
        pool_size=5,  
        pool_timeout=30,  
        pool_recycle=-1  
    )
    # 把所有被Base管理的表都删除
    Base.metadata.drop_all(engine)

if __name__ == '__main__':
    init_db()
    # drop_db()
"""
直接右击执行文件即可创建/删除 表和字段
"""
(1)单表操作
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker
from models import User

engine = create_engine(
    "mysql+pymysql://root:111@127.0.0.1:3306/aaa",
    max_overflow=0,
    pool_size=5
)
Connection = sessionmaker(bind=engine)

# 每次执行数据库操作时,都需要创建一个Connection,是从连接池中直接拿链接
conn = Connection()

######## 执行ORM操作########
# 创建出对象,通过conn.add(对象),把对象增加进去
obj1 = User(name='cc')
conn.add(obj1)
# 提交事务
conn.commit()
# 关闭session,其实是将连接放回连接池
conn.close()
(2)一对多的关系,新增数据

方式一:

conn = Connection()
# 先在hobby表中准备数据
hobby = Hobby(caption='橄榄球')

person = Person(name='cc', hobby_id=1)  # hobby_id不支持直接传hobby对象
conn.add_all([hobby, person])

conn.commit()
conn.close()

方式二:

conn = Connection()
# 先在hobby中准备两条数据
hobby = Hobby(caption='橄榄球')

person = Person(name='cc', hobby=hobby)  # 通过person表的hobby外键关系传hobby对象
conn.add_all([hobby, person])

conn.commit()
conn.close()

三、scoped_session线程安全

flask中已经集成了scoped_session,如果单独使用sqlalchemy,需要处理一下线程安全问题

示例:

from sqlalchemy.orm import sessionmaker
from sqlalchemy import create_engine
from sqlalchemy.orm import scoped_session
from models import User

engine = create_engine("mysql+pymysql://root:111@127.0.0.1:3306/aaa", max_overflow=0, pool_size=5)

conn = sessionmaker(bind=engine)
session = scoped_session(conn)

"""
# 线程安全,基于本地线程实现每个线程用同一个session
# 特殊的:因为scoped_session类实例化的时候会执行顶级代码:
	for meth in Session.public_methods:
    	setattr(scoped_session, meth, instrument(meth))
# 从而将原来Session类中的以下方法映射到scoped_session中:
public_methods = (
    '__contains__', '__iter__', 'add', 'add_all', 'begin', 'begin_nested',
    'close', 'commit', 'connection', 'delete', 'execute', 'expire',
    'expire_all', 'expunge', 'expunge_all', 'flush', 'get_bind',
    'is_modified', 'bulk_save_objects', 'bulk_insert_mappings',
    'bulk_update_mappings',
    'merge', 'query', 'refresh', 'rollback',
    'scalar'
)
"""
obj1 = User(name='jason')
session.add(obj1)

session.commit()
session.close()

四、基本增删查改

from sqlalchemy.orm import sessionmaker, scoped_session
from sqlalchemy import create_engine
from sqlalchemy.sql import text
from models import User, Person, Hobby

engine = create_engine(
    "mysql+pymysql://root:111@127.0.0.1:3306/aaa",
    max_overflow=0,
    pool_size=5
)

conn = sessionmaker(engine)
session = scoped_session(conn)

查询

res=session.query(Users).all()    # 查所有
res=session.query(Users).all()[0] # 取第一个
res=session.query(Users).first()  # 取第一个

# 查询时,等式用filter_by, 不等式(比较)用filter
# filter传的是表达式,filter_by传的是参数
r1 = session.query(User).filter(User.id < 3)  
# 没有加.all() 或者.first(),结果就是原生sql
print(r1) # SELECT users.id AS users_id, users.name AS users_name, users.email AS users_email, users.ctime AS users_ctime, users.extra AS users_extra FROM users WHERE users.id < %(id_1)s

# 查询Users表中id<3的记录
r2 = session.query(User).filter(User.id < 3).all()
print(r2)  # 结果是一个列表
for user in r2:
    print(user.name)

# 查询Users表中id为3的第一条记录
r3 = session.query(User).filter_by(id=3).first()
print(r3.name)

删除

r4 = session.query(User).filter(User.id > 2).delete()
print(r4)  # 1  (结果为影响的记录条数)

r5 = session.query(User).filter_by(id=1).delete()

session.commit()
session.close()

修改

r6 = session.query(User).filter_by(id=1).update({"name": 'cc'})
print(res)  # 结果为影响的记录条数
# 类似于django的F查询
# synchronize_session=False 表示加字符串
r7 = session.query(User).filter(User.id > 0).update({"name": User.name + '_xxx'}, synchronize_session=False)
# synchronize_session="evaluate" 表示数学运算
r8 = session.query(User).filter(User.id > 0).update({"age": User.age + 1}, synchronize_session='evaluate')
# 与上面的结果一样
r9 = session.query(User).filter(User.id > 0).update({User.age: User.age + 1}, synchronize_session='evaluate')

session.commit()
session.close()

as别名

r10 = ssion.query(User.name.label('xxx'), User.age).all()

for user in r10
    print(user.xxx)  # users表的 name字段 在代码中使用就必须使用 xxx,用name就会报错

:占位符,用params传参数

# select * from user where id<6 and name=cc order by id;
r11 = session.query(User).filter(text("id<:value and name=:name")).params(value=6, name='cc').order_by(User.id).all()
print(r11)

#自定义查询sql
r12 = session.query(User).from_statement(text("SELECT * FROM users where name=:name")).params(name='cc').all()
for u in r12:
    print(u.id)

其他查询操作

# in_ (注意下划线)
r13 = session.query(User).filter(User.id.in_([1,3,4])).all()

# ~非  (取反)
r14 = session.query(User).filter(~User.id.in_([1,3,4])).all()
print(ret)

# 二次筛选
r15 = session.query(User).filter(User.id.in_(session.query(User.id).filter_by(name='cc'))).all()

from sqlalchemy import and_, or_
# or_包裹的都是or条件,and_包裹的都是and条件
r16 = session.query(User).filter(and_(User.id > 3, User.name == 'cc')).all()
r17 = session.query(User).filter(User.id > 3, User.name == 'cc').all()  # and条件的and_()可以省略
r18 = session.query(User).filter(or_(User.id < 5, User.name == 'cc')).all()

r19 = session.query(User).filter(
    or_(
        User.id < 2,
        and_(User.name == 'eric', User.id > 3),
        User.age != 8
    ))
print(r19)

通配符

# 以c开头
r20 = session.query(User).filter(User.name.like('c%')).all()
# 不包含c
r21 = session.query(User).filter(~User.name.like('%c%')).all()

限制(用于分页、区间)

# select * from users limit 1,4;   从第一条数据往后取4条
r22 = session.query(User)[0:4]
for user in r22:
    print(user.age)

排序

# 按照name字段降序排
r23 = session.query(User).order_by(User.name.desc()).all()
for user in r23:
    print(user.id)

#第一个条件重复后,再按第二个条件升序排
r24 = session.query(User).order_by(User.name.desc(), User.id.asc()).all()

分组

from sqlalchemy.sql import func

# select max(id),sum(id),min(id) from users group by age;
r25 = session.query(
    func.max(User.id),
    func.sum(User.id),
    func.min(User.id)).group_by(User.age).all()
print(r25)

having筛选

# select max(id),sum(id),min(id) from user group by name having min(id)>2;

ret = session.query(
    func.max(User.id),
    func.sum(User.id),
    func.min(User.id)).group_by(User.name).having(func.min(User.id) > 2).all()

# 相当于django中的orm:
# User.objects.value(User.name).filter().annotate(a=max(User.id),b=min(User.id)).filter(b__gt=2).all()

连表(默认使用forinkey关联)

#join表,默认是inner join,没有指定on的字段,默认用外键关联
# select * from Person inner join Hobby on  person.hobby_id =hobby.id;
ret = session.query(Person).join(Hobby)
print(ret)

# isouter=True 外连,表示Person left join Hobby,没有右连接,反过来即可
ret = session.query(Person).join(Hobby, isouter=True)
ret = session.query(Hobby).join(Person, isouter=True)
print(ret)

# 自己指定on条件(连表条件),第二个参数,支持on多个条件,用and_,同上
# select * from Person left join Hobby on person.id=hobby.id
ret = session.query(Person).join(Hobby,Person.hobby_id==Hobby.id, isouter=True).all()
print(ret)
# select * from Person inner join Hobby on Person.hobby_id=Hobby.id where person.id>1 and hobby.caption='篮球'
ret = session.query(Person).join(Hobby).filter(Person.id>1,Hobby.caption=='篮球').all()

UNION组合(了解)

# UNION 操作符用于合并两个或多个 SELECT 语句的结果集
# union和union all的区别?————union去重,union all 不去重

q1 = session.query(Person.name).filter(Person.nid > 2)
q2 = session.query(Hobby.caption).filter(Hobby.id < 2)
ret = q1.union(q2).all()
print(ret)

q1 = session.query(Person.name).filter(Person.nid > 2)
q2 = session.query(Hobby.caption).filter(Hobby.id < 2)
ret = q1.union_all(q2).all()
print(ret)

五、多对多关系的建立和操作

1、models.py

import datetime
from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import relationship
from sqlalchemy import Column, Integer, String, Text, ForeignKey, DateTime, UniqueConstraint, Index
Base = declarative_base()


class Boy2Girl(Base):
    __tablename__ = 'boy2girl'
    id = Column(Integer, primary_key=True, autoincrement=True)
    girl_id = Column(Integer, ForeignKey('girl.id'))
    boy_id = Column(Integer, ForeignKey('boy.id'))


class Girl(Base):
    __tablename__ = 'girl'
    id = Column(Integer, primary_key=True)
    name = Column(String(64), unique=True, nullable=False)


class Boy(Base):
    __tablename__ = 'boy'

    id = Column(Integer, primary_key=True, autoincrement=True)
    name = Column(String(64), unique=True, nullable=False)

    # 与生成表结构无关,仅用于查询方便,放在哪个单表中都可以
    girls = relationship('Girl', secondary='boy2girl', backref='boys')



engine = create_engine(
        "mysql+pymysql://root:123@127.0.0.1:3306/aaa?charset=utf8",
        max_overflow=0,  # 超过连接池大小外最多创建的连接
        pool_size=5,  # 连接池大小
        pool_timeout=30,  # 池中没有线程最多等待的时间,否则报错
        pool_recycle=-1  # 多久之后对线程池中的线程进行一次连接的回收(重置)
    )

Base.metadata.create_all(engine)

2、基本增删查改

from sqlalchemy.orm import sessionmaker
from sqlalchemy import create_engine
from sqlalchemy.orm import scoped_session
from models import Boy, Girl, Boy2Girl
from sqlalchemy.sql import text

engine = create_engine("mysql+pymysql://root:123@127.0.0.1:3306/aaa", max_overflow=0, pool_size=5)
# 从连接池中拿一个链接
conn = sessionmaker(bind=engine)
session = scoped_session(conn)

boy1 = Boy(name='铁蛋')
boy2 = Boy(name='大壮')

girl1 = Girl(name='刘亦菲')
girl2 = Girl(name='迪丽热巴')
session.add_all([boy1, boy2, girl1, girl2])

# 建立关系
res = Boy2Girl(girl_id=1, boy_id=1)
# res = Boy2Girl(girl=Girl对象,boy=Boy对象)
session.add(res)

td = session.query(Boy).filter(Boy.id == 1).first()
dlrb = session.query(Girl).filter(Girl.id == 2).first()
lyf = session.query(Girl).filter(Girl.id == 1).first()

dz.girls=[dlrb,]
td.girls.append(lyf)

dz = session.query(Boy).filter(Boy.name == "大壮").first()
lyf = session.query(Girl).filter(Girl.name == "刘亦菲").first()
lyf.boys.append(dz)
session.add(lyf)

# print(lyf.boys)

session.commit()
session.close()

六、flask-sqlalchemy的使用

pip3 install flask-sqlalchemy

from flask import Flask
from flask_sqlalchemy import SQLAlchemy

app = Flask(__name__)
app.config["SQLALCHEMY_DATABASE_URI"] = "sqlite:///example.sqlite"
db = SQLAlchemy(app)
# db对象中有Column、Integer、String、session等属性,不用专门导入

# db.Model就是Base基表
class User(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String, unique=True, nullable=False)
    email = db.Column(db.String, unique=True, nullable=False)

db.session.add(User(username="cc", email="123@qq.com"))
db.session.commit()

users = User.query.all()

七、项目分文件使用flask-sqlalchemy + flask-migrate

# flask和SQLAlchemy的管理者,通过db把他们做连接
db = SQLAlchemy(app)
	- 包含配置
	- 包含ORM基类
	- 包含create_all()
	- engine
	- 创建连接
# pip3 install flask-migrate
# 安装了flask-migrate模块以后,就可以在flask项目中执行命令来操作数据库的初始化和迁移记录:
python3 manage.py db init     # 初始化:只执行一次
python3 manage.py db migrate  # 等同于 django中的makemigartions
python3 manage.py db upgrade  # 等同于 django中的migrate

补充:

django中的迁移记录可以指定app名,就会只迁移该app的记录
	- python3 manage.py makemigrations app01
	- python3 manage.py migrate app01