0. 前言

  • Marshmallow 是一个用于将 ORM 对象与 Python 原生数据类型之间转换的库。实现 object → dict、object → list、string → dict 和 string → list 等功能

1. Schema

  • 实现一个 object 和 json 之间的转化需要一个 Schema 对象作为中间载体,同时实现校验的功能:
class ImageTpl(Schema):
value = fields.Dict(required=True, error_messages=get_field_valid_msg("图片模版值"))
height = fields.Int(required=True, error_messages=get_field_valid_msg("图片模版高度"))
width = fields.Int(required=True, error_messages=get_field_valid_msg("图片模版宽度"))

2. 序列化

  • 序列化使用 dump() 或者 dumps() 方法,其中 dump() 实现 object → dict,dumps() 实现 object → string:
from marshmallow import pprint

user = User(name="Monty", email="monty@python.org")
schema = UserSchema()
result = schema.dump(user)
pprint(result.data)
# {"name": "Monty",
# "email": "monty@python.org",
# "created_at": "2014-08-17T14:54:16.049594+00:00"}
  • 序列化的结果可以通过 only 参数指定字段:
from pprint import pprint

user_data = {
'created_at': '2014-08-11T05:26:03.869245',
'email': u'ken@yahoo.com',
'name': u'Ken'
}
schema = UserSchema()
result = schema.load(user_data)
pprint(result.data)
# {'name': 'Ken',
# 'email': 'ken@yahoo.com',
# 'created_at': datetime.datetime(2014, 8, 11, 5, 26, 3, 869245)},

3. 反序列化

  • 反序列化使用 load() 或者 loads() 方法,分别实现 dict → object 和 string → object。其中 dict → object 需要添加装饰器,自己实现逻辑:
from marshmallow import Schema, fields, post_load

class UserSchema(Schema):
name = fields.Str()
email = fields.Email()
created_at = fields.DateTime() @post_load
def make_user(self, data):
return User(**data) user_data = {
'name': 'Ronnie',
'email': 'ronnie@stones.com'
}
schema = UserSchema()
result = schema.load(user_data)
result.data # => <User(name='Ronnie')>

4. Field 对象

  • Schema 对象为每个属性赋值为一个 Field 对象设定转换类型的校验参数,具体如下:
  • validate 参数:指定一个 lambda 函数或者函数,定义校验逻辑,传入函数定义了 ValidationError 的话,返回信息会记录抛出的异常:
from marshmallow import Schema, fields, ValidationError

def validate_quantity(n):
if n < 0:
raise ValidationError('Quantity must be greater than 0.')
if n > 30:
raise ValidationError('Quantity must not be greater than 30.') class ItemSchema(Schema):
quantity = fields.Integer(validate=validate_quantity) in_data = {'quantity': 31}
result, errors = ItemSchema().load(in_data)
errors # => {'quantity': ['Quantity must not be greater than 30.']}
  • required 参数:标记该字段必须传递切被校验
  • error_messages 参数:传递字典定义错误返回信息:
def get_field_valid_msg(field_name):
return {
'required': '[%s] 字段必填' % field_name,
'type': '[%s] 字段类型不合法' % field_name, # used by Unmarshaller
'null': '[%s] 字段不能为空' % field_name,
'validator_failed': '[%s] 字段值不合法' % field_name
}
  • many 参数:同 fields.Nested 同用表示指定元素类型的数组类型,严格按照数据类型校验,fields.Nested 表示制定另一个 Schema 作为外键:
......
'components': fields.Nested(Component, required=True, many=True,
error_messages=get_field_valid_msg("组件信息")),
......
  • fields.List 类型:一类 Field 元素,可以指定元素类型,对于传递的非数组类型的元素,会自动包装为一个数组

5. 验证

  • 对于 Schema 的校验有很多方式,通过上述 Field 元素的相关参数已经实现了很多对元素的校验
  • 也可以通过定义 @validates(field_name) 装饰器定义特定属性的校验函数
  • 也可以通过定义 @validate_schema() 装饰器定义 Schema 级别的校验函数:
@validates_schema
def validate_element_type(self, value): switch = {
DpaVideoPackageElementType.TEXT : lambda x : x in DpaVideoPackageDataType.text_types(),
DpaVideoPackageElementType.IMAGE : lambda x : x in DpaVideoPackageDataType.image_types(),
DpaVideoPackageElementType.LOGO : lambda x : x in DpaVideoPackageDataType.logo_types()
} try:
if not switch[value['element_type']](value['data_type']):
raise ValidationError("组件类型和组件数据类型不匹配")
except KeyError as e:
raise ValidationError("组件类型不存在")
  • 对于一个 Schema,load() 和 loads 方法会在返回值中加入验证错误的信息:
class ImageOrImageMeta(Schema):
value = fields.String(required=True, error_messages=get_field_valid_msg("图片或图片元数据值"))
height = fields.Int(required=True, error_messages=get_field_valid_msg("图片或图片元数据高度"))
width = fields.Int(required=True, error_messages=get_field_valid_msg("图片或图片元数据宽度")) class ImageTpl(Schema):
value = fields.Dict(required=True, error_messages=get_field_valid_msg("图片模版值"))
height = fields.Int(required=True, error_messages=get_field_valid_msg("图片模版高度"))
width = fields.Int(required=True, error_messages=get_field_valid_msg("图片模版宽度")) class TextOrTextMeta(Schema):
value = fields.String(required=True, error_messages=get_field_valid_msg("文本或文本元数据值"))
default_val = fields.String(error_messages=get_field_valid_msg("文本或文本元数据默认值"))
max_length = fields.Int(required=True, error_messages=get_field_valid_msg("文本或文本元数据最大值"))
......
switch = {
DpaVideoPackageDataType.IMAGE : lambda x : ImageOrImageMeta().load(x),
DpaVideoPackageDataType.IMAGE_META : lambda x : ImageOrImageMeta().load(x),
DpaVideoPackageDataType.IMAGE_TPL : lambda x : ImageTpl().load(x),
DpaVideoPackageDataType.TEXT : lambda x : TextOrTextMeta().load(x),
DpaVideoPackageDataType.TEXT_META : lambda x : TextOrTextMeta().load(x),
} result = switch[value['data_type']](value['data'])
if result.errors:
raise ValidationError(result.errors)

4. 参考文献

Python(二)Marshmallow 库相关学习的更多相关文章

  1. Python关于PIL库的学习总结与成果展示

    一.关于PIL库的学习总结 PIL(Python Image Library)库是Python语言的第三方库,需要通过pip工具安装.安装PIL库的方法如下,需要注意,安装库的名字是pillow. : ...

  2. python 协程库gevent学习--gevent数据结构及实战(三)

    gevent学习系列第三章,前面两章分析了大量常用几个函数的源码以及实现原理.这一章重点偏向实战了,按照官方给出的gevent学习指南,我将依次分析官方给出的7个数据结构.以及给出几个相应使用他们的例 ...

  3. MySQL数据库(二)--库相关操作、表相关操作(1)、存储引擎、数据类型

    一.库相关操作 1.创建数据库 (1)语法 create database 数据库 charset utf8; (2)数据库命名规范 可以由字母.数字.下划线.@.#.$ 区分大小写 唯一性 不能使用 ...

  4. MySQL数据库(二)——库相关操作、表相关操作(一)、存储引擎、数据类型

    库相关操作.表相关操作(一).存储引擎.数据类型 一.库相关操作 1.创建数据库 (1)语法 create database 数据库 charset utf8; (2)数据库命名规范 可以由字母.数字 ...

  5. python 协程库gevent学习--gevent源码学习(二)

    在进行gevent源码学习一分析之后,我还对两个比较核心的问题抱有疑问: 1. gevent.Greenlet.join()以及他的list版本joinall()的原理和使用. 2. 关于在使用mon ...

  6. MySql学习笔记【二、库相关操作】

    命令规范 关键字.函数名称大写 数据库.表名.字段名小写 语句须以分号结尾 切换使用数据库 USE database_name 如:USE test 查看数据库列表 SHOW {DATABASES|S ...

  7. python calendar标准库基础学习

    # -*- coding: utf-8 -*-# 作者:新手__author__ = 'Administrator'#标准库:日期时间基础学习:calendar:处理日期#例1import calen ...

  8. [python]-数据科学库Numpy学习

    一.Numpy简介: Python中用列表(list)保存一组值,可以用来当作数组使用,不过由于列表的元素可以是任何对象,因此列表中所保存的是对象的指针.这样为了保存一个简单的[1,2,3],需要有3 ...

  9. python 协程库gevent学习--gevent数据结构及实战(四)

    一不留神已经到第四部分了,这一部分继续总结数据结构和常用的gevent类,废话不多说继续. 1.Timeout错误类 晚上在调试调用第三方接口的时候,发现有些接口耗时非常多,觉得应该有个超时接口来限制 ...

随机推荐

  1. WPF布局原则

    WPF系统使用基于流布局的布局标准,开发人员创建与显示分辨率和窗口大小无关的用户界面.在不同显示器上可以进行很好的缩放. 首先来谈一谈布局原则: WPF窗口只能包含一个元素(Window元素属于内容控 ...

  2. TCP的三次握手过程?为什么会采用三次握手,若采用二次握手可以吗

    谢希仁版<计算机网络>中的例子: "已失效的连接请求报文段”的产生在这样一种情况下: client发出的第一个连接请求报文段并没有丢失,而是在某个网络结点长时间的滞留了,以致延误 ...

  3. Python 绘图与可视化 matplotlib text 与transform

    Text 为plots添加文本或者公式,反正就是添加文本了 参考链接:https://matplotlib.org/api/_as_gen/matplotlib.pyplot.text.html#ma ...

  4. dapperDemo

    dapperDemo 下载链接:http://pan.baidu.com/s/1geQHXPT

  5. nginx 反向代理时丢失端口的解决方案(转)

    今天,配置nginx反向代理时遇到一个问题,当设置nginx监听80端口时转发请求没有问题.但一旦设置为监听其他端口,就一直跳转不正常:如,访问欢迎页面时应该是重定向到登录页面,在这个重定向的过程中端 ...

  6. express和express-generator搭建web应用

    nodemon的安装使用 安装 npm i nodemon -D 配置 新建nodemon.json { "watch": "src/**/*.* } 修改package ...

  7. Android viewPager+fragment实现滑页效果

    先上图,手指在手机向左或者向右滑就可以实现相应的页面切换. 先看activity_main.xml文件,非常简单,主要是三个标题TextView和viewpager <?xml version= ...

  8. zookeeper中的QuorumPeerMain解析

    https://www.cnblogs.com/7758521gorden/p/8006983.html zookeeper中的QuorumPeerMain解析   在一个初级的hadoop与zook ...

  9. udp,select超时和recvfrom收不到数据原因

    wirshark抓包,发现有数据.但是select超时,直接recvfrom又失败. 代码中需要改进:select超时后,会移除fd_set集合中超时的那个句柄,所以每次要重新进行FD_SET,然后再 ...

  10. libpcap工具包使用go交叉编译开发android

    命令使用 libpcap交叉编译 cd /tmpwget http://www.tcpdump.org/release/libpcap-1.8.1.tar.gztar xvf libpcap-1.8. ...