20201028-第八讲-SQLAlchemy -orm

SQLAlchemy -orm

ORM:Object Relationship Mapping,对象关系映射,通过ORM可以通过类的方式操作数据库,不用写原生的SQL语句。通过把表映射成类,把行作为实例,把字段作为属性,ORM执行对象操作时最终会把对应的操作转换为数据库原生语句

使用ORM优点

易用性:有效减少SQL语句,写出的模型更直观

性能损耗小

设计灵活:可以轻松写出来复杂的查询

可移植性:SQLAlchemy 封装了底层的数据库实现,支持多个关系型数据库,包括MySQL,SQLite

使用SQLAlchemy

要使用ORM来操作数据库,先需要创建一个类来与对应的表进行映射。以User表来做例子,他有自增长的id、name、fullname、password这些字段,那么对应的类为

from sqlalchemy import Column,Integer,String

from constants import DB_URI

from sqlalchemy import create_engine

from sqlalchemy.ext.declarative import declarative_base

engine = create_engine(DB_URI,echo=True)

# 所有的类都要继承自`declarative_base`这个函数生成的基类

Base = declarative_base(engine)

class User(Base):

    # 定义表名为users

    __tablename__ = 'users'

    # 将id设置为主键,并且默认是自增长的

    id = Column(Integer,primary_key=True)

    # name字段,字符类型,最大的长度是50个字符

    name = Column(String(50))

    fullname = Column(String(50))

    password = Column(String(100))

    # 让打印出来的数据更好看,可选的

    def __repr__(self):

        return "<User(id='%s',name='%s',fullname='%s',password='%s')>" % (self.id,self.name,self.fullname,self.password)

SQLAlchemy 会自动设置第一个Integer 主键并且没有被标记为外键字段添加自增长的属性。因此以上例子中id自动的变成自增长的。以上创建完和表映射的类后,还没有真正的映射到数据库当中,执行以下代码将类映射到数据库只能中

1.Base.metadata.create_all()

在创建完数据表,且做完和数据库映射后,接下来让我们添加数据进去

ed_user = User(name='ed',fullname='Ed Jones',password='edspassword')

# 打印名字

print(ed_user.name)

# 打印密码

print(ed_user.password)

# 打印id

print(ed_user.id)

上例看出,name、password都能正常打印,唯独id为None,这是因为id 是一个自增长的主键,还未插入到数据库中,id 是不存在的。接下来让我们把创建的数据插入到数据库中,和数据库到交道的,是一个叫做Session的对象

from sqlalchemy.orm import sessionmaker

Session = sessionmaker(bind=engine)

# 或者

# Session = sessionmaker()

# Session.configure(bind=engine)

session = Session()

ed_user = User(name='ed',fullname='Ed Jones',password='edspassword')

session.add(ed_user)

现在只是把数据添加到session中,但并没有真正把数据存储到数据库中。如需要把数据库存储到数据库中,还要做一次commit操作

session.commit()

# 打印ed_user的id

print(ed_user.id)

这时,ed_user已经有id,说明已经插入数据库中。为什么添加到session中后还要做一次commit操作呢,因为,在SQLAlchemy 的ORM实现中,在做commit 操作前,所有操作都在事务中进行的,因此如将事务中的操作真正映射到数据库中,还需要做commit操作。既然是用到了事务,护腰并不能避免提到一个回滚操作了,那么看以下代码展示了如何使用回滚

# 修改ed_user的用户名

ed_user.name = 'Edwardo'

# 创建一个新的用户

fake_user = User(name='fakeuser',fullname='Invalid',password='12345')

# 将新创建的fake_user添加到session中

session.add(fake_user)

# 判断`fake_user`是否在`session`中存在

print(fake_user in session)

# 从数据库中查找name=Edwardo的用户

tmp_user = session.query(User).filter_by(name='Edwardo')

# 打印tmp_user的name

print(tmp_user)

# 打印出查找到的tmp_user对象,注意这个对象的name属性已经在事务中被修改为Edwardo了。

> <User(name='Edwardo', fullname='Ed Jones', password='edspassword')>

# 刚刚所有的操作都是在事务中进行的,现在来做回滚操作

session.rollback()

# 再打印tmp_user

print(tmp_user)

# 再看fake_user是否还在session中

print(fake_user in session)

接下来看如何进行查找操作,通过 session.query()方法实现的,这个方法返回一个 Query对象, Query对象相当于一个数组,装在查找出来的数据,并且可以进行迭代。里面装的什么数据,要看 session.query()方法传的什么参数了,如只是传一个ORM的类名作为参数,那么提取出来的数据就是都是这个类的实例

for instance in session.query(User).order_by(User.id):

    print(instance)

如传递二个及以上对象,或传递的事ORM类的属性,那么查找出来的就是元祖

for instance in session.query(User.name):

    print(instance)

以及:

for instance in session.query(User.name,User.fullname):

    print(instance)

或者是:

for instance in session.query(User,User.name).all():

    print(instance)

还可以对查找的结果(Query)做切片操作

for instance in session.query(User).order_by(User.id)[1:3]

    instance

如想对结果进行过滤,可以使用filter_by和filter二个方法,二种方法都用来做过滤的,区别在于,filter_by是传入关键字参数,filter是传入条件判断,并且filter能够传入的条件更多更灵活

# 第一种:使用filter_by过滤:

for name in session.query(User.name).filter_by(fullname='Ed Jones'):

    print(name)

# 第二种:使用filter过滤:

for name in session.query(User.name).filter(User.fullname=='Ed Jones'):

    print(name)


使用orm创建表格

。连接数据库

。创建Base基类

。创建session

。添加创建实例添加数据

。添加数据到数据库

。提交数据

Flask数据库的增删改查

def add_data():

      user = User(添加属性的数据)

      session.add(user)

      sesssion.commit()

def search_data():

      data = session.query(Usre).filter(条件判断).all()

      print(data)

def update_data():

      data = session.query(Usre).filter(条件判断).all()

      data.属性名 = xxx

      session.commit()

def delete_data():

      data = session.query(Usre).filter(条件判断).all()

      session.rollback()

      session.delete(data)

sqlalchemy 常用数据类型

Integer:整形。

Float:浮点类型。

Boolean:传递True/False进去。

DECIMAL:定点类型。

enum:枚举类型。

Date:传递datetime.date()进去。 

Datetime: 传递datetime.datetime()进去。

Time:传递datetime.time()进去。

String:字符类型,使用时需要指定长度,区别于Text类型。

Text:文本类型。

LONGTEXT:长文本类型。

column常用参数和聚合函数

column常用参数

default:默认值。 当你设置了default=xxx的时候,你如果没有传递这个数据,那么数据库会显示你设置的这个值

nullable:是否可空。 nullable=False 表示不能为空

primary_key:是否为主键。

unique:是否唯一。 当你设置了这个参数的时候,那么统一列的数据不能相同,相同就报错

autoincrement:是否自动增长。

onupdate:更新的时候执行的函数。

name:该属性在数据库中的字段映射

query 可用参数

1.模型对象。指定查找这个模型中所有的对象。

2.模型中的属性。可以指定只查某个模型的其中几个属性。

3.聚合函数。

。func.count:统计行的数量。

。func.max:求最大值。

。func.min:求最小值。

。func.avg:求平均值。

。func.sum:求和。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354