zoukankan      html  css  js  c++  java
  • sqlalchemy

    一、SQLAlchemy

    SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作,简言之便是:将对象转换成SQL,然后使用数据API执行SQL并获取执行结果。

    安装

    pip3 install 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 
    

    创建表

    #!/usr/bin/env python
    # -*- coding:utf-8 -*-
    from sqlalchemy.ext.declarative import declarative_base
    from sqlalchemy import Column,Integer,String,ForeignKey,UniqueConstraint,Index
    from sqlalchemy.orm import sessionmaker,relationship
    from sqlalchemy import create_engine
    engine = create_engine('mysql+pymysql://root:534198lu@127.0.0.1:3306/com',echo = True)  # echo = True会把执行的过程显示出来
    Base = declarative_base()  #生成一个SQLORM基类
    class Son(Base):  #必须继承基类
        __tablename__ = 'son'  # 表名
        id = Column(Integer, primary_key=True) #各列的数据
        name = Column(String(32))
        age = Column(String(16))
    
    Base.metadata.create_all(engine)  # 创建表建构
    
    # #这两行触发sessionmaker类下的__call__方法,return得到 Session实例,赋给变量session,所以session可以调用Session类下的add,add_all等方法
    MySession = sessionmaker(bind=engine)
    session = MySession()    #用MySession对象进行基本的操作
    ed_user1 = User(name='xiaowang', fullname='Xiao Liu', password='111')
    ed_user2 = User(name='xiaoming', fullname='Xiao Liu', password='222')
    print(ed_user1,ed_user2)  #打印出来的是俩行数据的内存地址
    #session.add(ed_user1) #用于添加单行数据
    session.add_all([ed_user1,ed_user2])   #用于增加多行数据【第一行数据,第二行数据,......】
    session.commit()    #添加数据后必须提交
    插入数据
    session.query(User).filter(User.id ==1).delete()
    session.commit()
    删除数据
    session.query(Users).filter(Users.id > 2).update({"name" : "099"})
    session.query(Users).filter(Users.id > 2).update({Users.name: Users.name + "099"}, synchronize_session=False)
    session.query(Users).filter(Users.id > 2).update({"num": Users.num + 1}, synchronize_session="evaluate")
    session.commit()
    更新数据
    #获取所有数据,返回的是多条数据组成的列表
    print(session.query(User).all())
    #获取第一条数据,返回的是一条数据
    print(session.query(User).first())
    #limit ,相当于切片
    print(session.query(User).all()[1:3])
    #order_by 排序
    print(session.query(User).order_by(-User.id).all())
    #可以指定查询,all()返回的是列表套元祖,first()返回的只有元祖
    print(session.query(User.name,User.id).all())
    #获取某一列数据,类似于django的get,如果返回数据为多个则报错
    print(session.query(User).filter(User.id==5).one())
    #过滤数据,filter()括号内跟的是条件
    print(session.query(User).filter(User.name == 'xiaowang').first())
    #过滤数据,filter_by()括号内跟的是键值对
    print(session.query(User).filter_by(name = 'xiaowang').first())
    查询数据
    • 其他类型
    # 条件
    ret = session.query(Users).filter_by(name='alex').all()
    ret = session.query(Users).filter(Users.id > 1, Users.name == 'eric').all()
    ret = session.query(Users).filter(Users.id.between(1, 3), Users.name == 'eric').all()
    ret = session.query(Users).filter(Users.id.in_([1,3,4])).all()
    ret = session.query(Users).filter(~Users.id.in_([1,3,4])).all()
    ret = session.query(Users).filter(Users.id.in_(session.query(Users.id).filter_by(name='eric'))).all()
    from sqlalchemy import and_, or_
    ret = session.query(Users).filter(and_(Users.id > 3, Users.name == 'eric')).all()
    ret = session.query(Users).filter(or_(Users.id < 2, Users.name == 'eric')).all()
    ret = session.query(Users).filter(
        or_(
            Users.id < 2,
            and_(Users.name == 'eric', Users.id > 3),
            Users.extra != ""
        )).all()
    
    
    # 通配符
    ret = session.query(Users).filter(Users.name.like('e%')).all()
    ret = session.query(Users).filter(~Users.name.like('e%')).all()
    
    # 限制
    ret = session.query(Users)[1:2]
    
    # 排序
    ret = session.query(Users).order_by(Users.name.desc()).all()
    ret = session.query(Users).order_by(Users.name.desc(), Users.id.asc()).all()
    
    # 分组
    from sqlalchemy.sql import func
    
    ret = session.query(Users).group_by(Users.extra).all()
    ret = session.query(
        func.max(Users.id),
        func.sum(Users.id),
        func.min(Users.id)).group_by(Users.name).all()
    
    ret = session.query(
        func.max(Users.id),
        func.sum(Users.id),
        func.min(Users.id)).group_by(Users.name).having(func.min(Users.id) >2).all()
    
    # 连表
    
    ret = session.query(Users, Favor).filter(Users.id == Favor.nid).all()
    
    ret = session.query(Person).join(Favor).all()
    
    ret = session.query(Person).join(Favor, isouter=True).all()
    
    
    # 组合
    q1 = session.query(Users.name).filter(Users.id > 2)
    q2 = session.query(Favor.caption).filter(Favor.nid < 2)
    ret = q1.union(q2).all()
    
    q1 = session.query(Users.name).filter(Users.id > 2)
    q2 = session.query(Favor.caption).filter(Favor.nid < 2)
    ret = q1.union_all(q2).all()
    其他查询方式

    一对多,多对多表的创建

    #!/usr/bin/env python
    # -*- coding:utf-8 -*-
    from sqlalchemy.ext.declarative import declarative_base
    from sqlalchemy import Column, Integer, String, ForeignKey, UniqueConstraint, Index
    from sqlalchemy.orm import sessionmaker, relationship
    from sqlalchemy import create_engine
    engine = create_engine('mysql+pymysql://root:534198lu@127.0.0.1:3306/com')#加echo = True显示执行过程
    Base = declarative_base()
    # 创建单表
    class Users(Base):
        __tablename__ = 'users'
        id = Column(Integer, primary_key=True)
        name = Column(String(32))
        extra = Column(String(16))
    
        __table_args__ = (    #组合索引
        UniqueConstraint('id', 'name', name='uix_id_name'),
            Index( 'ix_id_name','name', 'extra'),
        )
    
    # 一对多
    class Favor(Base):
        __tablename__ = 'favor'
        nid = Column(Integer, primary_key=True)
        caption = Column(String(50), default='red', unique=True)
    class Person(Base):
        __tablename__ = 'person'
        nid = Column(Integer, primary_key=True)
        name = Column(String(32), index=True, nullable=True)
        favor_id = Column(Integer, ForeignKey("favor.nid"))
    
    # # 多对多
    class Group(Base):
        __tablename__ = 'group'
        id = Column(Integer, primary_key=True)
        name = Column(String(64), unique=True, nullable=False)
        port = Column(Integer, default=22)
    
    class Server(Base):
        __tablename__ = 'server'
        id = Column(Integer, primary_key=True, autoincrement=True)
        hostname = Column(String(64), unique=True, nullable=False)
    
    class ServerToGroup(Base):
        __tablename__ = 'servertogroup'
        nid = Column(Integer, primary_key=True, autoincrement=True)
        server_id = Column(Integer, ForeignKey('server.id'))
        group_id = Column(Integer, ForeignKey('group.id'))
    
    def init_db():
        Base.metadata.create_all(engine)
    
    def drop_db():
      Base.metadata.drop_all(engine)
    一对多,多对多

    son = relationship('Son',backref='father')  用于建立俩张表之间的关系

    #!/usr/bin/env python
    # -*- coding:utf-8 -*-
    from sqlalchemy import create_engine
    from sqlalchemy import Column, Integer, String, ForeignKey
    from sqlalchemy.ext.declarative import declarative_base
    from sqlalchemy.orm import sessionmaker, relationship
    
    engine = create_engine('mysql+pymysql://root:534198lu@127.0.0.1:3306/com')#加echo = True显示执行过程
    Base = declarative_base()
    
    
    class Student(Base):
        __tablename__ = 'student'
        nid = Column(Integer, primary_key=True, autoincrement=True)
        name = Column(String(20))
        cour= relationship('Course', backref='student')
        #建立俩者之间的关系,其中backref='student'相当于在下面的类中写studeng = relationship('Student')效果一样
    
    
    class Course(Base):
        __tablename__ = 'course'
        nid = Column(Integer, primary_key=True, autoincrement=True)
        course = Column(String(20))
        student_id = Column(Integer, ForeignKey('student.nid'))
        # studeng = relationship('Student')
    
    Session = sessionmaker(bind=engine)
    session = Session()
    
    Base.metadata.create_all(engine)
    # Base.metadata.drop_all(engine)
    #
    s1 = Student(name = 'zane')
    c1 = Course(course='wuli')
    c2 = Course(course='huaxue')
    c3 = Course(course='shuxue')
    s1.cour = [c1,c2,c3]   #批量添加,相当于是c1,c2,c3 共同和s1有关系
    session.add_all([s1,c1,c2,c3])
    session.commit()
    
    c1 = session.query(Course).filter(Course.student_id==6).first()
    print(c1)
    print(c1.student)   #通过c1.student就可以获取和c1有对应关系的Student表中所有的数据
    print(c1.student.name)
    # s1 = session.query(Student).filter(Student.name == 'zane').first()
    # print(s1)
    # print(s1.cour)
    # print(s1.cour[0].course)
    # print(s1.cour[1].course)
    relationship俩种用法

    print( session.query(Course.course,Student.name).join(Student).all())------------用于关联表的查询

    son = relationship('Son',backref='father')内部和join一样也是一个关联查询

    s1.cour = [c1,c2,c3] 用于给子表重新建立对应关系,如果表里面已经有对应关系,这样做的话会把里面的对应冲掉,

    如果是新添加数据,又需要建立对应关系的话用s1.cour.append(w3)

    ForeignKey建在多的一方

    多对多实例:(注意第三张表一定要放到最上面)

    #!/usr/bin/env python
    # -*- coding:utf-8 -*-
    from sqlalchemy import create_engine
    from sqlalchemy import Column, Integer, String, ForeignKey
    from sqlalchemy.ext.declarative import declarative_base
    from sqlalchemy.orm import sessionmaker, relationship
    
    engine = create_engine('mysql+pymysql://root:534198lu@127.0.0.1:3306/com?charset=utf8')#加echo = True显示执行过程
    Base = declarative_base()
    
    class Man_to_Wemon(Base):
        __tablename__ = 'Men_to_Wemon'
        nid = Column(Integer, primary_key=True, autoincrement=True)
        man_id = Column(Integer,ForeignKey('man.id'))
        women_id= Column(Integer,ForeignKey('women.id'))
    
    class Man(Base):
        __tablename__ = 'man'
        id = Column(Integer, primary_key=True, autoincrement=True)
        name = Column(String(20))
        age = Column(String(16))
    
    class Women(Base):
        __tablename__ = 'women'
        id = Column(Integer, primary_key=True, autoincrement=True)
        name = Column(String(20))
        age = Column(String(16))
        bf = relationship('Man',secondary =  Man_to_Wemon.__table__,backref = 'gf')
    #secondary =  Man_to_Wemon.__table__ 对应第三张表(使用与有第三张表的情况)
    
    Base.metadata.create_all(engine)
    
    Session = sessionmaker(bind=engine)
    session = Session()
    
    # m1 = Man(name = 'alex',age = '18')
    # m2 = Man(name = 'tony',age = '18')
    # w1 = Women(name = '如花',age = '18')
    # w2 = Women(name = '铁锤',age = '18')
    #普通的添加数据
    # session.add_all([m1,m2,w1,w2])
    # session.commit()
    # t1 = Man_to_Wemon(man_id=1,women_id=2)
    # session.add_all([t1])
    # session.commit()
    
    
    #利用relationship往关系表中添加数据
    m1 = session.query(Man).filter_by(id=2).first()   #获取一个男人
    w1 = session.query(Women).all()   #获取好几个妹纸
    m1.gf = w1  #建立起一个男人对应好几个妹纸的一个对应关系
    session.add(m1)
    session.commit()
    多对多
    # #利用relationship往关系表中添加数据
    m1 = session.query(Man).filter_by(id=1).first()   #获取一个男人
    print(m1)
    w1 = session.query(Women).filter_by(id=1).first()  #获取好几个妹纸
    print(w1)
    m1.gf =[w1]  #建立起一个男人对应好几个妹纸的一个对应关系
    print(m1.gf)
    session.add(m1)
    session.commit()
    

     注意:

    m1.gf =[w1] 中如果m1.gf以前有值的时候,再m1.gf =[w1]这样的话就会把其中的内容从新赋值,把以前的对应关系替换掉

    m1.gf =[w1] 中如果等号后面的w1获取的就是一个列表的时候直接写m1.gf =w1就可以了,如果w1只是一个单独的值的话就应该放到列表中

  • 相关阅读:
    从一个word文件中读取所有的表格和标题(1)
    多线程下QAxObject指针为NULL的解决办法
    event对象,ie8及其以下
    日期插件kalendae,遇到的一些问题
    回车键和button按钮都绑定同一个事件,如何避免按回车的时候button重复点击
    jQuery.Cookie.js用法
    jquery操作radio单选按钮、checked复选框。
    拖拽改变div的大小
    [BZOJ 2242] [SDOI 2011] 计算器
    20181016提高测试
  • 原文地址:https://www.cnblogs.com/luxiaojun/p/5752408.html
Copyright © 2011-2022 走看看