转自:数据管道

作者:言淦,一枚喜欢淦代码的码农,"言淦说"主理人

背景

其实一开始用的是pymysql,但是发现维护比较麻烦,还存在代码注入的风险,所以就干脆直接用ORM框架。

ORM即Object Relational Mapper,可以简单理解为数据库表和Python类之间的映射,通过操作Python类,可以间接操作数据库。

Python的ORM框架比较出名的是SQLAlchemyPeewee,这里不做比较,只是单纯讲解个人对SQLAlchemy的一些使用,希望能给各位朋友带来帮助。

  • sqlalchemy版本: 1.3.15

  • pymysql版本: 0.9.3

  • mysql版本: 5.7

初始化工作

一般使用ORM框架,都会有一些初始化工作,比如数据库连接,定义基础映射等。

以MySQL为例,创建数据库连接只需要传入DSN字符串即可。其中echo表示是否输出对应的sql语句,对调试比较有帮助。

from sqlalchemy import create_engineengine = create_engine('mysql+pymysql://$user:$password@$host:$port/$db?charset=utf8mb4', echo=True)

个人设计

对于我个人而言,引进ORM框架时,我的项目会参考MVC模式做以下设计。其中model存储的是一些数据库模型,即数据库表映射的Python类;model_op存储的是每个模型对应的操作,即增删查改;调用方(如main.py)执行数据库操作时,只需要调用model_op层,并不用关心model层,从而实现解耦。

├── main.py
├── model
│   ├── __init__.py
│   ├── base_model.py
│   ├── ddl.sql
│   └── py_orm_model.py
└── model_op├── __init__.py└── py_orm_model_op.py

映射声明(Model介绍)

举个栗子,如果我们有这样一张测试表

create table py_orm (`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '唯一id',`name` varchar(255) NOT NULL DEFAULT '' COMMENT '名称',`attr` JSON NOT NULL COMMENT '属性',`ct` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',`ut` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON update CURRENT_TIMESTAMP COMMENT '更新时间',PRIMARY KEY(`id`)
)ENGINE=InnoDB COMMENT '测试表';

在ORM框架中,映射的结果就是下文这个Python类

# py_orm_model.py
from .base_model import Base
from sqlalchemy import Column, Integer, String, TIMESTAMP, text, JSONclass PyOrmModel(Base):__tablename__ = 'py_orm'id = Column(Integer, autoincrement=True, primary_key=True, comment='唯一id')name = Column(String(255), nullable=False, default='', comment='名称')attr = Column(JSON, nullable=False, comment='属性')ct = Column(TIMESTAMP, nullable=False, server_default=text('CURRENT_TIMESTAMP'), comment='创建时间')ut = Column(TIMESTAMP, nullable=False, server_default=text('CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP'), comment='更新时间')

首先,我们可以看到PyOrmModel继承了Base类,该类是sqlalchemy提供的一个基类,会对我们声明的Python类做一些检查,我将其放在base_model中。

# base_model.py
# 一般base_model做的都是一些初始化的工作from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_baseBase = declarative_base()engine = create_engine("mysql+pymysql://root:123456@127.0.0.1:33306/orm_test?charset=utf8mb4", echo=False)

其次,每个Python类都必须包含__tablename__属性,不然无法找到对应的表。

第三,关于数据表的创建有两种方式,第一种当然是手动在MySQL中创建,只要你的Python类定义没有问题,就可以正常操作;第二种是通过orm框架创建,比如下面

# main.py
# 注意这里的导入路径,Base创建表时会寻找继承它的子类,如果路径不对,则无法创建成功from sqlachlemy_lab import Base, engineif __name__ == '__main__':Base.metadata.create_all(engine)

创建效果:

...
2020-04-04 10:12:53,974 INFO sqlalchemy.engine.base.Engine
CREATE TABLE py_orm (id INTEGER NOT NULL AUTO_INCREMENT, name VARCHAR(255) NOT NULL DEFAULT '' COMMENT '名称', attr JSON NOT NULL COMMENT '属性', ct TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, ut TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, PRIMARY KEY (id)
)

第四,关于字段属性

  • 1.primary_key和autoincrement比较好理解,就是MySQL的主键和递增属性。

  • 2.如果是int类型,不需要指定长度,而如果是varchar类型,则必须指定。

  • 3.nullable对应的就是MySQL中的NULL 和 NOT NULL

  • 4.关于defaultserver_default: default代表的是ORM框架层面的默认值,即插入的时候如果该字段未赋值,则会使用我们定义的默认值;server_default代表的是数据库层面的默认值,即DDL语句中的default关键字。

Session介绍

在SQLAlchemy的文档中提到,数据库的增删查改是通过session来执行的。

>>> from sqlalchemy.orm import sessionmaker
>>> Session = sessionmaker(bind=engine)>>> session = Session()>>> orm = PyOrmModel(id=1, name='test', attr={})
>>> session.add(orm)>>> session.commit()>>> session.close()

如上,我们可以看到,对于每一次操作,我们都需要对session进行获取,提交和释放。这样未免过于冗余和麻烦,所以我们一般会进行一层封装。

1.采用上下文管理器的方式,处理session的异常回滚和关闭,这部分与所参考的文章是几乎一致的。

# base_model.py
from contextlib import contextmanager
from sqlalchemy.orm import sessionmaker, scoped_sessiondef _get_session():"""获取session"""return scoped_session(sessionmaker(bind=engine, expire_on_commit=False))()# 在这里对session进行统一管理,包括获取,提交,回滚和关闭
@contextmanager
def db_session(commit=True):session = _get_session()try:yield sessionif commit:session.commit()except Exception as e:session.rollback()raise efinally:if session:session.close()

2.在PyOrmModel中增加两个方法,用于model和dict之间的转换

class PyOrmModel(Base):...@staticmethoddef fields():return ['id', 'name', 'attr']@staticmethoddef to_json(model):fields = PyOrmModel.fields()json_data = {}for field in fields:json_data[field] = model.__getattribute__(field)return json_data@staticmethoddef from_json(data: dict):fields = PyOrmModel.fields()model = PyOrmModel()for field in fields:if field in data:model.__setattr__(field, data[field])return model

3.数据库操作的封装,与参考的文章不同,我是直接调用了session,从而使调用方不需要关注model层,减少耦合。

# py_orm_model_op.py
from sqlachlemy_lab.model import db_session
from sqlachlemy_lab.model import PyOrmModelclass PyOrmModelOp:def __init__(self):pass@staticmethoddef save_data(data: dict):with db_session() as session:model = PyOrmModel.from_json(data)session.add(model)# 查询操作,不需要commit@staticmethoddef query_data(pid: int):data_list = []with db_session(commit=False) as session:data = session.query(PyOrmModel).filter(PyOrmModel.id == pid)for d in data:data_list.append(PyOrmModel.to_json(d))return data_list

4.调用方

# main.py
from sqlachlemy_lab.model_op import PyOrmModelOpif __name__ == '__main__':PyOrmModelOp.save_data({'id': 1, 'name': 'test', 'attr': {}})

完整代码请参见:

https://github.com/yangancode/python_lab/tree/master/sqlachlemy_lab

这是我开发的机器人公众号小号,目前增加了天气查询,955公司名单,关注时间查询;后面还会增加图片功能和每日送书抽奖送书活动,以及调戏功能,欢迎来体验,捧场

一个机器人公众号已经上线,欢迎调戏

推荐阅读:
入门: 最全的零基础学Python的问题  | 零基础学了8个月的Python  | 实战项目 |学Python就是这条捷径
干货:爬取豆瓣短评,电影《后来的我们》 | 38年NBA最佳球员分析 |   从万众期待到口碑扑街!唐探3令人失望  | 笑看新倚天屠龙记 | 灯谜答题王 |用Python做个海量小姐姐素描图 |碟中谍这么火,我用机器学习做个迷你推荐系统电影
趣味:弹球游戏  | 九宫格  | 漂亮的花 | 两百行Python《天天酷跑》游戏!
AI: 会做诗的机器人 | 给图片上色 | 预测收入 | 碟中谍这么火,我用机器学习做个迷你推荐系统电影
小工具: Pdf转Word,轻松搞定表格和水印! | 一键把html网页保存为pdf!|  再见PDF提取收费! | 用90行代码打造最强PDF转换器,word、PPT、excel、markdown、html一键转换 | 制作一款钉钉低价机票提示器! |60行代码做了一个语音壁纸切换器天天看小姐姐!|

年度爆款文案

  • 1).卧槽!Pdf转Word用Python轻松搞定!

  • 2).学Python真香!我用100行代码做了个网站,帮人PS旅行图片,赚个鸡腿吃

  • 3).首播过亿,火爆全网,我分析了《乘风破浪的姐姐》,发现了这些秘密

  • 4).80行代码!用Python做一个哆来A梦分身

  • 5).你必须掌握的20个python代码,短小精悍,用处无穷

  • 6).30个Python奇淫技巧集

  • 7).我总结的80页《菜鸟学Python精选干货.pdf》,都是干货

  • 8).再见Python!我要学Go了!2500字深度分析!

  • 9).发现一个舔狗福利!这个Python爬虫神器太爽了,自动下载妹子图片

点阅读原文,看B站我的视频!

一个超方便使用SQL的Python神器!相关推荐

  1. 一个超方便使用SQL的Python神器

    作者:言淦,一枚喜欢淦代码的码农,"言淦说"主理人 其实一开始用的是pymysql,但是发现维护比较麻烦,还存在代码注入的风险,所以就干脆直接用ORM框架. ORM即Object ...

  2. 神器推荐!一个能监控文件变化的Python神器—看门狗

    假设现在有一个应用场景,需要对文件系统进行监控,发生变化时产生日志,对新增的文件做一些相应的操作. 比如说应用到我们之前的音乐高潮提取器:若当前文件夹下增加了一个音乐文件,监控器就调用音乐高潮提取器, ...

  3. 一个超好用的论文图片编辑神器---Inkscape(Mark)

    好文配好图,结合我的科研实践和身边研究人员的使用频率,极力给小伙伴们推荐Inkscape用于矢量图编辑.让你得心应手画图,告别"科研能力强,配图能力弱"的窘境. 一.Inkscap ...

  4. 给大家分享一个私藏已久的Python神器!

    今天给大家推荐一个优质的Python公众号「法纳斯特」,作者:小F. 学习编程是一个比较枯燥的过程,所以小F平常喜欢分享一些有趣.有料的Python原创项目实战.从2018年8月一直到现在,已经更新接 ...

  5. 一个傻瓜式构建可视化 web的 Python 神器

    大家好,我是明哥. 之前给大家安利过一款构建可视化 web app 的 Python 工具库 pywebio,不知道有没有人去试用下? 今天要介绍这个神器,可以说是  pywebio  的 Plus  ...

  6. python(pygame)开发一个超简易版消灭病毒

    python(pygame)开发一个超简易版消灭病毒 消灭病毒游戏 以下是游戏代码:每个类都是一个py文件,需要导包 1.主程序代码 2.飞机类 3.子弹类 4.病毒类 消灭病毒游戏 消灭病毒是一个打 ...

  7. 一个傻瓜式构建可视化 web的 Python 神器 ——streamlit

    之前给大家安利过一款构建可视化 web app 的 Python 工具库 pywebio,不知道有没有人去试用下? 今天要介绍这个神器,可以说是 pywebio 的 Plus + Pro + Max版 ...

  8. 卧槽!又一个Python神器!

    今天给大家推荐一个优质公众号「Python技术」,作者 派森酱 .这是我关注了很久的一个技术公众号,抱歉,现在才分享给大家.大家可以点击下方卡片关注: 学习编程是一个比较枯燥的过程,所以派森酱平常喜欢 ...

  9. SQL注入测试神器sqlmap

    点击上方蓝字"开源优测"一起玩耍 声明 本公众号所有内容,均属微信公众号: 开源优测  所有,任何媒体.网站或个人未经授权不得转载.链接.转贴或以其他方式复制发布/发表.已经本公众 ...

最新文章

  1. 计算机系统的组成doc,《计算机系统的组成》.doc
  2. 算法和编程面试题精选 TOP50!(附代码+解题思路+答案)
  3. 《软件需求分析(第二版)》第 17 章——需求管理工具 重点部分总结
  4. Kotlin 5 控制流,返回和跳转
  5. javascript二级省市select联动菜单
  6. 测试计划和测试方案有什么区别?
  7. html中div圆角效果,div+css实现圆角即网页上常用的圆角效果
  8. 如何在线免费caj转word
  9. 程序设计python_Python程序设计—车万翔
  10. 计算机管理有U盘 为啥不显示,U盘插入电脑后不显示怎么办?
  11. Android打开自启动设置页面
  12. 数据链路层------基于TCP/IP五层模型
  13. 低压差线性稳压器 LDO
  14. 用 C# picturebox 控件画图
  15. 彻底禁用停用杀死WIN10的自动更新
  16. 爬取自如网站杭州市的租房信息
  17. acw - 4262.空调
  18. Windows Oracle ODBC驱动数据源安装配置
  19. C#毕业设计——基于C#+asp.net+sqlserver的客户关系管理系统设计与实现(毕业论文+程序源码)——客户关系管理系统
  20. python调用ansys_windows下用python调用HFSS

热门文章

  1. 安卓性能优化—使用ArrayMap与SparseArray
  2. 10讲项目实战首页底部固定位置的分享、咨询功能
  3. 数据库编程和设计——JDBC技术
  4. oracle数据库 笛卡尔积,Oracle连接查询和笛卡尔积
  5. 骨传导蓝牙耳机听歌怎么样、音质最好的骨传导蓝牙耳机
  6. 运营商MCC+MNC
  7. Eclipse安装Sonar插件SonarLint
  8. 前端vue,nodejs + mysql(vue请求mysql接口数据)
  9. [Git]Git指南一 查看创建删除标签
  10. 人脸检测器:对DSFD的理解