Flask中的ORM - SQLAlchemy

# Django中默认使用sqlite,且提供了ORM框架通过类和对象操作数据库
# Flask当中没有ORM框架,需要引入SQLAlchemy扩展

# pip install flask-sqlalchemy 
# 连接的是mysql数据库需要安装
# pip install flask-mysqldb 

# 设置数据库链接地址:
    app.config['SQLALCHEMY_DATABASE_URI’] = mysql://root:chuanzhi@127.0.0.1:3306/flask_test
# 关闭自动追踪数据库:
    app.config[’SQLALCHEMY_TRACK_MODIFICATION’] = False


Flask 数据库设置:

名字备注
SQLALCHEMY_DATABASE_URI用于连接的数据库 URI 。例如:sqlite:tmp/test.dbmysql://username:password@server/db
SQLALCHEMY_BINDS一个映射 binds 到连接 URI 的字典。更多 binds 的信息见用 Binds 操作多个数据库
SQLALCHEMY_ECHO如果设置为Ture, SQLAlchemy 会记录所有 发给 stderr 的语句,这对调试有用。(打印sql语句)
SQLALCHEMY_RECORD_QUERIES可以用于显式地禁用或启用查询记录。查询记录 在调试或测试模式自动启用。更多信息见get_debug_queries()。
SQLALCHEMY_NATIVE_UNICODE可以用于显式禁用原生 unicode 支持。当使用 不合适的指定无编码的数据库默认值时,这对于 一些数据库适配器是必须的(比如 Ubuntu 上 某些版本的 PostgreSQL )。
SQLALCHEMY_POOL_SIZE数据库连接池的大小。默认是引擎默认值(通常 是 5 )
SQLALCHEMY_POOL_TIMEOUT设定连接池的连接超时时间。默认是 10 。
SQLALCHEMY_POOL_RECYCLE多少秒后自动回收连接。这对 MySQL 是必要的, 它默认移除闲置多于 8 小时的连接。注意如果 使用了 MySQL , Flask-SQLALchemy 自动设定 这个值为 2 小时。

SQLAlchemy 字段类型:
类型名python中类型说明
Integerint普通整数,一般是32位
SmallIntegerint取值范围小的整数,一般是16位
BigIntegerint或long不限制精度的整数
Floatfloat浮点数
Numericdecimal.Decimal普通整数,一般是32位
Stringstr变长字符串
Textstr变长字符串,对较长或不限长度的字符串做了优化
Unicodeunicode变长Unicode字符串
UnicodeTextunicode变长Unicode字符串,对较长或不限长度的字符串做了优化
Booleanbool布尔值
Datedatetime.date时间
Timedatetime.datetime日期和时间
LargeBinarystr二进制文件

# SQLAlchemy 列选项

选项名说明
primary_key如果为True,代表表的主键
unique如果为True,代表这列不允许出现重复的值
index如果为True,为这列创建索引,提高查询效率
nullable如果为True,允许有空值,如果为False,不允许有空值
default为这列定义默认值

# SQLAlchemy 关系选项

选项名说明
backref在关系的另一模型中添加反向引用
primary join明确指定两个模型之间使用的联结条件
uselist如果为False,不使用列表,而使用标量值
order_by指定关系中记录的排序方式
secondary指定多对多关系中关联表的名称
secondaryjoin在SQLAlchemy中无法自行决定时,指定多对多关系中的二级连接条件
 


# 定义自定义模型类
    class Role(db.Model):
        __tablename__ = ‘role’
        id = db.Column(db.Integer, primary_key=True)
        name = db.Column(db.String(64), nullable=False, unique=True)

        # lazy, 默认为subquery,自动加载数据库数据,dynamic反之
        users = db.relationship(‘Users’, backref=‘roles’, lazy=‘dynamic')
    


# 查询所有用户数据
User.query.all() -> 返回所有的对象,列表显示, 不是queryset

查询有多少个用户
User.query.count()

查询第1个用户
User.query.first()

查询id为4的用户[3种方式]
User.query.get(4)  -> Flask中只能根据id来查,如果查不到不会报错
User.query.filter_by(id=4).first()  -> 精确查询
User.query.filter(User.id == 4).first()  -> 模糊查询, >= 

查询名字结尾字符为g的所有数据[开始/包含]
User.query.filter(User.name.endswith('g')).all()
User.query.filter(User.name.startswith('z')).all()

查询名字不等于wang的所有数据[2种方式]
User.query.filter(User.name != 'wang').all()

from sqlalchemy import not_
User.query.filter(not_(User.name == "wang")).all()

查询名字和邮箱都以 li 开头的所有数据[2种方式]
User.query.filter(User.name.startswith('li'), User.email.startswith('li')).all()

# 查询password是 `123456` 或者 `email` 以 `itheima.com` 结尾的所有数据
from sqlalchemy import or_
User.query.filter(or_(User.password == '123456', User.email.endswith('itheima.com'))).all()

查询id为 [1, 3, 5, 7, 9] 的用户列表
User.query.filter(User.id.in_([1, 3, 5, 7, 9])).all()

# 查询所有用户数据,并以邮箱排序
User.query.order_by(User.email).all()  # 默认升序
User.query.order_by(User.email.desc()).all()

每页3个,查询第2页的数据
    # 返回的是pagination对象,如果查不到会返回False
pagination = User.query.paginate(2, 3)
pagination.items  # 当前页对象
pagination.pages  # 总页数
pagination.page   # 当前页页数

# 关联查询

 
 
 
# 关联属性users,但是不是列
# backref='role' 代表给User增加属性role 通过user.role可以获得关联属性信息
# lazy 类似于惰性查询,默认subquery,会自动查询关联集对象,dynamic则不会    
users = db.relationship('User', backref='role', lazy='subquery')
# 查询admin角色的所有用户信息
Role.query.filter(Role.name == 'admin').first().users

# 查询id为1 的用户的角色信息
User.query.get(1).role

# lazy = dynamic

In [2]: ro1 = Role.query.get(1)

In [3]: ro1.users
Out[3]: <sqlalchemy.orm.dynamic.AppenderBaseQuery at 0x101f3a510>

In [4]: ro1.users[0]
Out[4]: <User: wang wang@163.com 123456 1>


# 数据库的迁移: 避免直接删除数据库,可以通过框架追踪数据库变化
    # pip install flask-migrate

    # manage.add_command(‘db’, MigrateCommand)
    # Migrate(app, db)

# coding=utf-8
from flask import Flask, render_template
from flask_sqlalchemy import SQLAlchemy
from flask_migrate import Migrate, MigrateCommand
from flask_script import Manager



app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'mysql://root:chuanzhi@127.0.0.1:3306/flask_migrate'
# 动态追踪数据库
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False


db = SQLAlchemy(app)

# 通过script 以脚本的方式运行服务器
manager = Manager(app)

# 给脚本加入db 指定集
manager.add_command('db', MigrateCommand)

# 创建Migrate类,传入app和db,以迁移方式管理数据库
Migrate(app, db)

class Role(db.Model):
    __tablename__ = 'roles'
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(64), nullable=False, unique=True)

    def __repr__(self):
        return '<Role: id=%s, name=%s>' % (self.id, self.name)


@app.route('/')
def index():
    return 'index'

if __name__ == '__main__':
    # app.run(debug=True)
    manager.run()

# 操作指令
    # python manage.py db init 进行数据库管理初始化工作,生成迁移文件夹

    # python manage.py db migrate -m '说明信息' 生成当前版本的迁移文件,仍未操作数据库

    # python manage.py db upgrade 迁移生成数据库中的表
    # python manage.py db upgrade/downgrade <version_num> 升级或者降级到指定版本

    # python manage.py db history 查看数据库迁移历史

    # python manage.py db current 查看当前数据库版本














  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值