JSON.parse() 的实现
1. JSON.parse()
JSON.parser() 是JSON 格式的一个函数, 它用于将object 数据类型转换成为JSON 数据类型, 这里我们来自己实现一下JSON.parser() 函数.
2. 前置知识
2.1 JSON格式中的数据类型
JSON 格式中, 可以将需要处理数据类型分为以下6类, 注意这里的意思是需要处理的属性类型有以下6 类, 表示的是数据处理的6 种情况;
真实的数据分类并不是按以下分类的, 这里需要注意.
- 字符串
- 数字
- 布尔值
- null 值
- 符号
- 包含转义字符的字符串(字符串和包含转义字符的字符串处理方式不同)
2.2 转义字符的处理
2.2 判断对象是否相等
自己实现一个函数, 用于判断两个对象是否相等, 实现代码如下:
// 判断两个对象是否相等的函数
const objectEquals = (a, b) => {
// 如果 a, b相等,则返回 true,否则返回 false
// Object.getOwnPropertyNames()方法返回一个由指定对象的所有自身属性的属性名(包括不可枚举属性但不包括Symbol值作为名称的属性)组成的数组
// 换句话来说 Object.getOwnPropertyNames()方法返回的是对象所有 key 组成的数组 list
var aProps = Object.getOwnPropertyNames(a)
var bProps = Object.getOwnPropertyNames(b)
if (aProps.length != bProps.length) {
return false
}
for (var i = 0; i < aProps.length; i++) {
var propName = aProps[i]
if (typeof a[propName] === 'object') {
let judge = objectEquals(a[propName], b[propName])
if (!judge) {
return false
}
} else if (a[propName] !== b[propName]) {
return false
}
}
return true
}
2.3 寻找匹配的字符串
这里匹配的字符串主要是括号的匹配, 理解下面的代码
// 返回对象 '{' 对应的 '}',返回数组 '[' 对应的 ']'
const getAimIndex = (string, index) => {
let str = string
// breakStr是结束判断字符
let startStr = str[index]
let breakStr = startStr === '{' ? '}' : ']'
let judgeNum = 0
/*
判断逻辑:
1. 遍历从 i = 1 开始
2. if temp = '{' || '[', judgeNum++
3. if temp = '}' || ']', judgeNum--
4. if temp = '}' || ']' && judgeNum === 1, return i
*/
for (let i = index, len = str.length; i < len; i++) {
let temp = str[i]
if (temp === startStr) {
judgeNum++
} else if (temp === breakStr && judgeNum !== 1) {
judgeNum--
} else if (temp === breakStr && judgeNum === 1) {
return i
}
}
// log('judgeNum: ', judgeNum)
}
let str = '{{{}}}'
console.log(getAimIndex(str, 0))
2.4 基础的递归思想
这个实现过程中, 很多内容的实现都需要递归去完成, 因此对于递归有基本的了解会比较好理解.
理解过程可能不会很费劲, 但是如果自己实现的话, 个人认为如果能够将 Title2.2 判断对象是否相等 独立实现就可以自己实现JSON.parse()
3. 实现流程
3.1 将JSON 字符串解析成为tokens 数组
在生产tokens 数组中, 直观上, 我们会认为我们只需要将JSON 中有意义的字符串添加在tokens 数组中即可, 但是, 这样做会存在一个问题:
对于如下代码
{
"obj1": true,
"obj2": "true"
}
上面代码中, obj1 obj2 属性本质上是不一样的, 一个是String 类型, 一个是Boolean 类型, 而我们从JSON 中读取的内容类型均为String 类型; 因此为了确保完美解析JSON 数据, 我们必须在tokens 数组中存储JSON 数据的值类型
同时, 由于我们生成对象的时候必须依赖值的类型, 值本身去生成对象, 因此需要提供一个方法返回值内容
综上所述, 我们使用面向对象的思想来解决这个问题:
const log = console.log.bind(console)
const TYPE = {
"[object Number]": 1,
"[object String]": 2,
"[object Null]": 3,
"[object Boolean]": 4,
"character": 5,
"escapeCharater": 6
}
const includeEscapeCharacter = function(str) {
let asciiEscape = [0, 39, 34, 92, 10, 13, 11, 9, 8, 12]
for (let i = 0, len = str.length; i < len; i++) {
let temp = str[i].charCodeAt(0)
if (asciiEscape.indexOf(temp) !== -1) {
return true
}
}
return false
}
const dealEscape = (str) => {
let escape = {
b: `\b`,
f: '\f',
n: '\n',
r: '\r',
t: '\t',
'\\': '\\',
'\/': '\/',
'\"': '\"',
"\'": "\'"
}
let result = ''
let string = str
let i = 0
let len = string.length
// log(str)
while (i < len) {
let temp = string[i]
if (temp === '\\') {
let endIndex = i + 1
result += escape[string[endIndex]]
i = endIndex + 1
} else {
result += temp
i++
}
}
return result
}
const getType = function (value) {
if (value === ':' || value === '{' || value === '}' || value === '[' || value === ']') {
return 5
}
if (includeEscapeCharacter(value)) {
return 6
}
let type = Object.prototype.toString.apply(value)
return TYPE[type]
}
class JsonValue {
constructor(value) {
this.value = String(value)
this.type = getType(value)
}
toString() {
if (this.type === 1) {
return Number(this.value)
} else if (this.type === 2) {
return String(this.value)
} else if (this.type === 3) {
return null
} else if (this.type === 4) {
return this.value === 'true'
} else if (this.type === 5) {
return String(this.value)
} else if (this.type === 6) {
return dealEscape(this.value)
}
}
}
const __main = () => {
let a = new JsonValue('a')
let b = new JsonValue(1)
log(a)
log(a.toString())
log(b)
log(b.toString())
}
__main()
tokens 数组中存储的都是JsonValue 类型的数据;
接下来遍历JSON 数据添加内容即可
3.2 将tokens 数组拼接成为Object 对象
这个过程就是遍历tokens 数组, 依照符号(:)的索引index, tokens[index - 1] 为key, tokens[index + 1] 为value; 找到key-value 对, 然后添加到结果即可; 需要注意的是这个过程需要考虑Object 的嵌套, 以及Array中Object 的嵌套, 这里需要使用递归去处理
这个过程的核心代码如下: 根据不同的情况, 处理不同的内容
// 获取key-value对的value值
const getTokensValue = (index, array) => {
let judge = array[index].value
if (judge === '{') {
let nextIndex = getAimIndex(array, index)
let sliceList = array.slice(index + 1, nextIndex)
return parsedDict(sliceList)
} else if (judge === '[') {
let nextIndex = getAimIndex(array, index)
let sliceList = array.slice(index + 1, nextIndex)
return conversionObj(sliceList)
} else {
return array[index].toString()
}
}
4. code
代码地址: JSON.parse() 的实现
JSON.parse() 的实现的更多相关文章
- JSON.parse()和JSON.stringify()
1.parse 用于从一个字符串中解析出json 对象.例如 var str='{"name":"cpf","age":"23&q ...
- JSON.stringify()与JSON.parse()
JSON.stringify()用于把一个对象解析成字符串,如 var student = { age: 23, name: 'wang' } JSON.stringify(student); 结果: ...
- JSON.parse 与 eval() 对于解析json的问题
1.eval()与JSOn.parse的不同 eval() var c = 1; //全局变量 var jsonstr1 = '{"name":"a",&quo ...
- JSON.parse与eval的区别
JSON.parse与eval和能将一个字符串解析成一个JSON对象,但还是有挺大区别. 测试代码 var A = "{ a: 1 , b : 'hello' }"; var B ...
- ajex请求的数据 什么时候需用Json.parse()
ajex请求的数据 什么时候需用 Json.parse()
- JSON.parse和eval的区别
JSON.parse和eval的区别 JSON(JavaScript Object Notation)是一种轻量级的数据格式,采用完全独立于语言的文本格式,是理想的数据交换格式.同时,JSON是Jav ...
- JSON.stringify()和JSON.parse()
parse用于从一个字符串中解析出json对象,如 var str = '{"name":"huangxiaojian","age":&qu ...
- JSON.parse()和JSON.stringify()区别
parse用于从一个字符串中解析出json对象,如: var str = '{"name":"huangxiaojian","age":&q ...
- JSON.stringify() / JSON.parse()
JSON.stringify() 这个方法可以把javascript对象转换成json字符串. JSON.parse() 这个方法可以把 json 字符串转换成 javascript对象. [下面来看 ...
- js中解析json对象:JSON.parse()用于从一个字符串中解析出json对象, JSON.stringify()用于从一个对象解析出字符串。
JSON.parse()用于从一个字符串中解析出json对象. var str = '{"name":"huangxiaojian","age&quo ...
随机推荐
- keepalived高可用服务配置管理
实验环境: 主机 ipaddress 服务 备注 k8s-master1 10.0.0.63 nginx k8s-master2 10.0.0.64 nginx k8s-node1 10.0.0.65 ...
- lammps 学习之:系统压力太大,导致原子丢失
体系压力太大:146981.52bar,体系压强太大 会把原子挤跑 出现原子丢失的情况(lost atoms). 原子丢失: 解决方法:增大体系体积.增加z方向的距离.
- 【Spark】部署流程的深度了解
文章目录 Spark核心组件 Driver Executor Spark通用运行流程图 Standalone模式运行机制 Client模式流程图 Cluster模式流程图 On-Yarn模式运行机制 ...
- 【Spark】SparkStreaming的容错机制
文章目录 检查点机制 驱动器程序容错 工作节点容错 接收器容错 处理保证 检查点机制 Metadata checkpointing -- 将定义流计算的信息存入容错的系统如HDFS. Data che ...
- 【poj 2406】Power Strings 后缀数组DC3模板 【连续重复子串】
Power Strings 题意 给出一个字符串s,求s最多由几个相同的字符串重复而成(最小循环节的重复次数) 思路 之前学习KMP的时候做过. 我的思路是:枚举字符串的长度,对于当前长度k,判断\( ...
- 使用Optional,不再头疼NPE
前言 在 Java 语言开发中,可能大多数程序员遇到最多的异常就是 NullPointException 空指针异常了.这个当初语言的开发者"仅仅因为这样实现起来更容易"而允许空引 ...
- EOS基础全家桶(十一)智能合约IDE-EOS_Studio
简介 我们马上要进入智能合约的开发了,以太坊最初提供了智能合约的功能,并宣告区块链进入2.0时代,而EOS的智能合约更进一步,提供了更多的便利性和可能性.为了进一步了解智能合约,并进行开发,我们需要先 ...
- springBoot第二种配置文件yaml书写方式及读取数据、整合myBatis和整合junit
一.yaml文件格式:key-value形式:可以表示对象 集合 1.语法:key:value 冒号后面必须跟一个空格再写value值 key1: key2: key3:value 2.属性取值:a. ...
- 2018-06-19 js DOM对象
DOM对象: Doucument Object Model即文档对象 DOM对象的操作: 1.找元素 返回元素对象: var obj=document.getElementById();//通过Id查 ...
- 浅谈HTTP和HTTPS
HTTP和HTTPS协议 网络协议:计算机之间为了实现网络通信而达成的一种“约定”或“规则”,有了这种“约定”,不同厂商的生产设备,以及不同不同操作系统组成的计算机之间,就可以实现通信. HTTP(H ...