Python实现JSON生成器和递归下降解释器
Python实现JSON生成器和递归下降解释器
github地址:https://github.com/EStormLynn/Python-JSON-Parser
目标
从零开始写一个JSON的解析器,特征如下:
- 符合标准的JSON解析器和生成器
- 手写递归下降的解释器(recursive descent parser)
- 使用Python语言(2.7)
- 解释器和生成器少于500行
- 使用cProfile完成性能分析和优化
实现内容
- [x] 解析字面量(true false null)
- [x] 解析数字
- [x] 解析字符串
- [x] 解析Unicode
- [x] 解析数组
- [x] 解析对象
- [x] 单元测试
- [x] 生成器
- [x] cProfile性能优化
详细介绍
JSON是什么
JSON(JavaScript Object Notation)是一个用于数据交换的文本格式,参考ecma标准,JSON Data Interchange Format,先看一段JSON的数据格式:
{
"title": "Design Patterns",
"subtitle": "Elements of Reusable Object-Oriented Software",
"author": [
"Erich Gamma",
"Richard Helm",
"Ralph Johnson",
"John Vlissides"
],
"year": 2009,
"weight": 1.8,
"hardcover": true,
"publisher": {
"Company": "Pearson Education",
"Country": "India"
},
"website": null
}
在json的树状结构中
- null: 表示为 null
- boolean: 表示为 true 或 false
- number: 一般的浮点数表示方式,在下一单元详细说明
- string: 表示为 "..."
- array: 表示为 [ ... ]
- object: 表示为 { ... }
实现解释器
es_parser 是一个手写的递归下降解析器(recursive descent parser)。由于 JSON 语法特别简单,可以将分词器(tokenizer)省略,直接检测下一个字符,便可以知道它是哪种类型的值,然后调用相关的分析函数。对于完整的 JSON 语法,跳过空白后,只需检测当前字符:
n ➔ literal
t ➔ true
f ➔ false
" ➔ string
0-9/- ➔ number
[ ➔ array
{ ➔ object
对于json的typevalue和json string编写了这样2个类
class EsValue(object):
__slots__ = ('type', 'num', 'str', 'array', 'obj')
def __init__(self):
self.type = JTYPE_UNKNOW
class context(object):
def __init__(self, jstr):
self.json = list(jstr)
self.pos = 0
以解析多余的空格,制表位,换行为例:
def es_parse_whitespace(context):
if not context.json:
return
pos = 0
while re.compile('[\s]+').match(context.json[pos]):
pos += 1
context.json = context.json[pos:]
解析字面量
字面量包括了false,true,null三种。
def es_parse_literal(context, literal, mytype):
e_value = EsValue()
if ''.join(context.json[context.pos:context.pos + len(literal)]) != literal:
raise MyException("PARSE_STATE_INVALID_VALUE, literal error")
e_value.type = mytype
context.json = context.json[context.pos + len(literal):]
return PARSE_STATE_OK, e_value
def es_parse_value(context, typevalue):
if context.json[context.pos] == 't':
return es_parse_literal(context, "true", JTYPE_TRUE)
if context.json[context.pos] == 'f':
return es_parse_literal(context, "false", JTYPE_FALSE)
if context.json[context.pos] == 'n':
return es_parse_literal(context, "null", JTYPE_NULL)
解析数字
JSON number类型,number 是以十进制表示,它主要由 4 部分顺序组成:负号、整数、小数、指数。只有整数是必需部分。
JSON 可使用科学记数法,指数部分由大写 E 或小写 e 开始,然后可有正负号,之后是一或多个数字(0-9)。
JSON 标准 ECMA-404 采用图的形式表示语法,可以更直观地看到解析时可能经过的路径:
python是一种动态语言,所以es_value中num可以是整数也可以是小数,
class es_value():
def __init__(self, type):
self.type = type
self.num = 0
python对于string类型,可以强制转换成float和int,但是int(string)无法处理科学记数法的情况,所以统一先转成float在转成int
typevalue.num = float(numstr)
if isint:
typevalue.num = int(typevalue.num)
实现的单元测试包含:
def testnum(self):
print("\n------------test number-----------")
self.assertEqual(type(self.parse("24")), type(1))
self.assertEqual(type(self.parse("1e4")), type(10000))
self.assertEqual(type(self.parse("-1.5")), type(-1.5))
self.assertEqual(type(self.parse("1.5e3")), type(1.500))
解析字符串
对于字符串中存在转义字符,在load的时候须要处理转义字符,\u的情况,进行编码成unicode
def es_parse_string(context):
charlist = {
'\\"': '\"',
"\\'": "\'",
"\\b": "\b",
"\\f": "\f",
"\\r": "\r",
"\\n": "\n",
"\\t": "\t",
"\\u": "u",
"\\\\": "\\",
"\\/": "/",
"\\a": "\a",
"\\v": "\v"
}
while context.json[pos] != '"':
# 处理转意字符
if context.json[pos] == '\\':
c = context.json[pos:pos + 2]
if c in charlist:
e_value.str += charlist[c]
else:
e_value.str += ''.join(context.json[pos])
pos += 1
continue
pos += 2
else:
e_value.str += ''.join(context.json[pos])
pos += 1
e_value.type = JTYPE_STRING
context.json = context.json[pos + 1:]
context.pos = 1
if '\u' in e_value.str:
e_value.str = e_value.str.encode('latin-1').decode('unicode_escape')
return PARSE_STATE_OK, e_value
单元测试:
def teststring(self):
print("\n------------test string----------")
self.assertEqual(type(self.parse("\" \\\\line1\\nline2 \"")), type("string")) # input \\ is \
self.assertEqual(type(self.parse("\" abc\\def\"")), type("string"))
self.assertEqual(type(self.parse("\" null\"")), type("string"))
self.assertEqual(type(self.parse("\"hello world!\"")), type("string"))
self.assertEqual(type(self.parse("\" \u751F\u5316\u5371\u673A \"")), type("string"))
es_dumps函数,json生成器
将python dict结构dumps成json串
def es_dumps(obj):
obj_str = ""
if isinstance(obj, bool):
if obj is True:
obj_str += "True"
else:
obj_str += "False"
elif obj is None:
obj_str += "null"
elif isinstance(obj, basestring):
for ch in obj.decode('utf-8'):
if u'\u4e00' <= ch <= u'\u9fff':
obj_str += "\"" + repr(obj.decode('UTF-8')) + "\""
break
else:
obj_str += "\"" + obj + "\""
elif isinstance(obj, list):
obj_str += '['
if len(obj):
for i in obj:
obj_str += es_dumps(i) + ", "
obj_str = obj_str[:-2]
obj_str += ']'
elif isinstance(obj, int) or isinstance(obj, float): # number
obj_str += str(obj)
elif isinstance(obj, dict):
obj_str += '{'
if len(obj):
for (k, v) in obj.items():
obj_str += es_dumps(k) + ": "
obj_str += es_dumps(v) + ", "
obj_str = obj_str[:-2]
obj_str += '}'
return obj_str
cProfile性能分析
导入cProfile模块进行性能分析,load中国34个省份地区人口发布,
import cProfile
from jsonparser import *
import json
cProfile.run("print(es_load(\"china.json\"))")
修改部分代码使用python build-in,优化context结构,string在copy的时候比list性能显著提高。消耗时间从20s降到1s
Python实现JSON生成器和递归下降解释器的更多相关文章
- Python 迭代器、生成器、递归、正则表达式 (四)
一.迭代器&生成器 1.迭代器仅仅是一容器对象,它实现了迭代器协议.它有两个基本方法: 1)next 方法 返回容器的下一个元素 2)_iter_方法 返回迭代器自身.迭代器可以使用内建的it ...
- Python技法:实现简单的递归下降Parser
1. 算术运算表达式求值 在上一篇博文<Python技法:用re模块实现简易tokenizer>中,我们介绍了用正则表达式来匹配对应的模式,以实现简单的分词器.然而,正则表达式不是万能的, ...
- Python入门之三元表达式\列表推导式\生成器表达式\递归匿名函数\内置函数
本章目录: 一.三元表达式.列表推导式.生成器表达式 二.递归调用和二分法 三.匿名函数 四.内置函数 ================================================ ...
- python高级之生成器&迭代器
python高级之生成器&迭代器 本机内容 概念梳理 容器 可迭代对象 迭代器 for循环内部实现 生成器 1.概念梳理 容器(container):多个元素组织在一起的数据结构 可迭代对象( ...
- 第三篇:python高级之生成器&迭代器
python高级之生成器&迭代器 python高级之生成器&迭代器 本机内容 概念梳理 容器 可迭代对象 迭代器 for循环内部实现 生成器 1.概念梳理 容器(container ...
- python中的生成器函数是如何工作的?
以下内容基于python3.4 1. python中的普通函数是怎么运行的? 当一个python函数在执行时,它会在相应的python栈帧上运行,栈帧表示程序运行时函数调用栈中的某一帧.想要获得某个函 ...
- Python三大器之生成器
Python三大器之生成器 生成器初识 什么是生成器 生成器本身属于迭代器.继承了迭代器的特性,惰性求值,占用内存空间极小. 为什么要有生成器 我们想使用迭代器本身惰性求值的特点创建出一个可以容纳百万 ...
- Python 迭代器和生成器(转)
Python 迭代器和生成器 在Python中,很多对象都是可以通过for语句来直接遍历的,例如list.string.dict等等,这些对象都可以被称为可迭代对象.至于说哪些对象是可以被迭代访问的, ...
- Python入门篇-生成器函数
Python入门篇-生成器函数 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.生成器概述 1>.生成器generator 生成器指的是生成器对象,可以由生成器表达式得到, ...
随机推荐
- 洛谷P3381 最小费用最大流模板
https://www.luogu.org/problem/P3381 题目描述 如题,给出一个网络图,以及其源点和汇点,每条边已知其最大流量和单位流量费用,求出其网络最大流和在最大流情况下的最小费用 ...
- 2017 ICPC西安区域赛 A - XOR (线段树并线性基)
链接:https://nanti.jisuanke.com/t/A1607 题面: Consider an array AA with n elements . Each of its eleme ...
- 内存泄漏之malloc替换方法
//内存泄漏之malloc替换方法 //内存泄漏之malloc替换方法#include "stdio.h"#include "stdlib.h" /*文件路径名 ...
- jdk1.8 -- 自定义FunctionInterface
一.自定义函数的理解 对于java提供的一些函数,用lambda表达式是可以解决一些问题的,但是在对于一些比较复杂的数据类型在处理时,可能会有些力不从心了,但是 我们可以通过自定义的一些函数,通过使用 ...
- 什么是SSL证书服务?
SSL证书服务(Alibaba Cloud SSL Certificates Service)由阿里云联合多家国内外数字证书管理和颁发的权威机构.在阿里云平台上直接提供的服务器数字证书.您可以在阿里云 ...
- shiro整合shiro多验证登录(账号密码登录和使用手机验证码登录)
1. 首先新建一个shiroConfig shiro的配置类,代码如下: @Configuration是标识这个类是一个配置文件,在启动时会加载这个类里面的内容,这个配置文件的位置的一定一定一定不能 ...
- 数据结构 -- 栈(Stack)
一.栈的简介 定义 栈(英语:stack)又称为堆栈或堆叠,栈作为一种数据结构,它按照先进后出的原则存储数据,先进入的数据被压入栈底,最后的数据在栈顶,需要读数据的时候从栈顶开始弹出数据(最后一个数据 ...
- java日志框架系列(3):logback框架配置详解
1.Logback配置 1.配置步骤及默认配置 logback即可以通过编程式配置,也可以通过xml的形式配置. logback配置步骤: 1. 尝试在 classpath 下查找文件 logback ...
- PHP获取今日、昨日、本周、上周、本月、上月、本季、上季、今年、去年
//今天开始$beginToday = date('Y-m-d 00:00:00', time());//今天结束$endToday = date('Y-m-d 23:59:59', time()); ...
- Redis学习存档(2)——通过Java使用Redis:Jedis
一.创建项目,引入jedis jar包 可在百度搜索maven repository 进入后搜索jedis,复制依赖包到pom.xml文件中 <project xmlns="http: ...