这一部分是对Date String Number Boolean扩展toString方法,Date的toString是返回UTC格式的字符串,而后面几个是返回原始值。

        function f(n) {// 返回两位数字字符串
return n < 10 ? '0' + n: n;
}
if (typeof Date.prototype.toJSON !== 'function') {//如果Date不支持原生的toJSON方法
Date.prototype.toJSON = function() {//扩展Date的toJSON方法
//是否是有穷数,如果为true,返回根据UTC时间计算出的年月日时分秒 YYYY-MM-DDThh:mm:ssZ如果为false,返回null
return isFinite(this.valueOf()) ? this.getUTCFullYear() + '-' + f(this.getUTCMonth() + 1) + '-' + f(this.getUTCDate()) +
'T' + f(this.getUTCHours()) + ':' + f(this.getUTCMinutes()) + ':' + f(this.getUTCSeconds()) + 'Z': null;
};
//扩展String Number Boolean的toJSON方法
String.prototype.toJSON = Number.prototype.toJSON = Boolean.prototype.toJSON = function() {
return this.valueOf();//返回他们的原始值
};
}

这里是扩展Stringify方法

if (typeof JSON.stringify !== 'function') {//扩展JSON的stringify方法
escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
meta = { // table of character substitutions
'\b': '\\b',
'\t': '\\t',
'\n': '\\n',
'\f': '\\f',
'\r': '\\r',
'"': '\\"',
'\\': '\\\\'
};
JSON.stringify = function(value, replacer, space) {//扩展JSON.stringify方法
var i;//for循环变量
gap = '';//分隔符
indent = '';//分隔符
// If the space parameter is a number, make an indent string containing that
// many spaces.
if (typeof space === 'number') {//如果是数值
for (i = 0; i < space; i += 1) {//分隔符为space个空格
indent += ' ';
} } else if (typeof space === 'string') {//如果是字符串
indent = space;//分隔符就是字符串
} rep = replacer;
if (replacer && typeof replacer !== 'function' && (typeof replacer !== 'object' || typeof replacer.length !== 'number')) {
throw new Error('JSON.stringify');//如果第二个参数存在,必须为函数或者数组(伪数组),否则抛出异常
} return str('', {
'': value
});
};
}

做了2件事:获取分隔符和replacer,内部调用str方法

        function str(key, holder) {//第一次调用时 key:'', holder:{'': value}
var i, // The loop counter.
k, // The member key.
v, // The member value.
length, mind = gap,//初始mind和gap都为""
partial, value = holder[key];//第二次调用时 value就是传入的键所对应的值
//如果value有toJSON方法
if (value && typeof value === 'object' && typeof value.toJSON === 'function') {
value = value.toJSON(key);//调用value.toJSON方法
} if (typeof rep === 'function') {//如果replace是一个方法
value = rep.call(holder, key, value);
}
// 判断value类型
switch (typeof value) {
case 'string'://如果是字符串,加引号
return quote(value);
case 'number'://如果是数值
//有穷数用原生的String()将数值转为符串,否则返回null
return isFinite(value) ? String(value) : 'null';
case 'boolean'://如果是bool值或者null,返回String(value)
case 'null':
return String(value);
case 'object'://如果是对象
if (!value) {//null
return 'null';
}
gap += indent;//分隔符
partial = [];//临时数组
if (Object.prototype.toString.apply(value) === '[object Array]') {//数组
length = value.length;
for (i = 0; i < length; i += 1) {//对数组的每一项递归调用str
partial[i] = str(i, value) || 'null';
}
// 如果partial为[],返回"[]"
// 如果 gap分隔符存在,返回[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']'
// 如果分隔符不存在,返回'[' + partial.join(',') + ']'
v = partial.length === 0 ? '[]': gap ? '[\n' + gap +
partial.join(',\n' + gap) + '\n' + mind + ']': '[' + partial.join(',') + ']';
gap = mind;//重置为""
return v;
}
if (rep && typeof rep === 'object') {//如果rep存在且为数组或者对象
length = rep.length;//如果是数组
for (i = 0; i < length; i += 1) {//过滤
if (typeof rep[i] === 'string') {
k = rep[i];//键是数组的值
v = str(k, value);//递归调用
if (v) {
//"key": value 或者"key":value
partial.push(quote(k) + (gap ? ': ': ':') + v);
}
}
}
} else {//如果不是数组或方法或不存在
for (k in value) {
if (Object.prototype.hasOwnProperty.call(value, k)) {
v = str(k, value);
if (v) {
partial.push(quote(k) + (gap ? ': ': ':') + v);
}
}
}
} v = partial.length === 0 ? '{}': gap ? '{\n' + gap +
partial.join(',\n' + gap) + '\n' + mind + '}': '{' + partial.join(',') + '}';
gap = mind;
return v;
}
}

下面是图解:

        function quote(string) {//将传入的字符串加上引号,有必要转义的先转义
escapable.lastIndex = 0;//起始位置从0开始
return escapable.test(string) ? '"' + string.replace(escapable,
function(a) {//匹配到的字符 如\t \n
var c = meta[a];//字符对应的转义表示
//如果c是字符串 ,直接返回对象中键所对应的值 '\t'=>'\\t'
//如果c不是字符串,也就是说它不在meta对象中,这时做不同的转义处理:
//拿 a为"\u0600"举例
//a.charCodeAt(0) =>1536
//a.charCodeAt(0).toString(16) => 600
//('0000' + a.charCodeAt(0).toString(16)) =>0000600
//('0000' + a.charCodeAt(0).toString(16)).slice( - 4) 取最后四位 => 0600
//'\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice( - 4)) '\\u0600'
return typeof c === 'string' ? c: '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice( - 4);
}) + '"': '"' + string + '"';
}

这里是扩展parse方法

        if (typeof JSON.parse !== 'function') {//如果JSON没有parse方法
cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
JSON.parse = function(text, reviver) {//扩展JSON方法
var j; function walk(holder, key) {// walk({'':j},'') var k, v, value = holder[key];//value第一次是传入的eval编译后的结果
if (value && typeof value === 'object') {//如果value存在并且是对象
for (k in value) {
if (Object.prototype.hasOwnProperty.call(value, k)) {//是否有原型上的属性
v = walk(value, k);//递归调用获取结果
if (v !== undefined) {
value[k] = v;
} else {
delete value[k];
}
}
}
}
//调用walk之前有判断,所以在这里reviver肯定存在
return reviver.call(holder, key, value);
}
text = String(text);
cx.lastIndex = 0;
if (cx.test(text)) {
text = text.replace(cx,
function(a) {
// \u0600 ---> \\u0600 因为\需要转义
return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice( - 4);
});
} // text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@')
// =>把\\t \\uffff 这类转为@
// =>var text='{"a":"\\t44","b":"\\uffff"}'; text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@'); //{"a":"@44","b":"@"}
// replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']')
// =>把非空字符串、数值、bool、null替换为]
// .replace(/(?:^|:|,)(?:\s*\[)+/g, '')
// => 把 [ ,[ :[ 这类的替换为''
// 如果剩余的字符串只剩下 ]:,{}和空格 就是测试通过,否则抛出异常
if (/^[\],:{}\s]*$/.test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@')
.replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { j = eval('(' + text + ')');//eval()传入的参数加括号编译 return typeof reviver === 'function' ? walk({//如果第二个参数是函数 返回walk({'':j},'')的结果,否则直接返回eval编译的结果
'': j
},
'') : j;
} throw new SyntaxError('JSON.parse');
};
}

通过一系列的替换操作,如果剩下的字符串只剩下 ]:,{}和空格,测试通过,接下来就可以用eval编译。

如果parse方法的第二个参数存在,返回walk的调用结果,否则直接返回eval编译结果。

//reviver的用法:
// var jsontext = '{ "hiredate": "2008-01-01T12:00:00Z", "birthdate": "2008-12-25T12:00:00Z" }';
// var dates = JSON.parse(jsontext, dateReviver);
// console.log(dates); // function dateReviver(key, value) {
// var a;
// if (typeof value === 'string') {
// a = /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z$/.exec(value);
// if (a) {
// return new Date(Date.UTC(+a[1], +a[2] - 1, +a[3], +a[4],
// +a[5], +a[6]));
// }
// }
// return value;
// };

json2.js 源码解读的更多相关文章

  1. js便签笔记(10) - 分享:json2.js源码解读笔记

    1. 如何理解“json” 首先应该意识到,json是一种数据转换格式,既然是个“格式”,就是个抽象的东西.它不是js对象,也不是字符串,它只是一种格式,一种规定而已. 这个格式规定了如何将js对象转 ...

  2. json2.js源码解读记录

    相关内容:json详细用法.js语法.unicode.正则   json特点--最简单.最小巧的经典js库.   json作者:道克拉斯.克劳福德(Douglas Crockford)--js大牛 出 ...

  3. js便签笔记(10) - 分享:json.js源码解读笔记

    1. 如何理解“json” 首先应该意识到,json是一种数据转换格式,既然是个“格式”,就是个抽象的东西.它不是js对象,也不是字符串,它只是一种格式,一种规定而已. 这个格式规定了如何将js对象转 ...

  4. fastclick.js源码解读分析

    阅读优秀的js插件和库源码,可以加深我们对web开发的理解和提高js能力,本人能力有限,只能粗略读懂一些小型插件,这里带来对fastclick源码的解读,望各位大神不吝指教~! fastclick诞生 ...

  5. prototype.js 源码解读(02)

    如果你想研究一些比较大型的js框架的源码的话,本人建议你从其最初的版本开始研读,因为最初的版本东西少,易于研究,而后的版本基本都是在其基础上不断扩充罢了,所以,接下来我不准备完全解读prototype ...

  6. prototype.js 源码解读(01)

    prototype.js是一个设计的非常优雅且很有实用价值的js基础类库,其源码非常值得研究.研究它的源码不仅能提升个人水平,而且对你打下坚实的js基础也很有帮助.因本人技术水平有限,该解读仅供参考. ...

  7. require.js 源码解读——配置默认上下文

    首先,我们先来简单说一下,require.js的原理: 1.载入模块
 2.通过模块名解析出模块信息,以及计算出URL
 3.通过创建SCRIPT的形式把模块加载到页面中.
 4.判断被加载的脚本,如 ...

  8. 亚马逊左侧菜单延迟z三角 jquery插件jquery.menu-aim.js源码解读

    关于亚马逊的左侧菜单延迟,之前一直不知道它的实现原理.梦神提到了z三角,我也不知道这是什么东西.13号那天很有空,等领导们签字完我就可以走了.下午的时候,找到了一篇博客:http://jayuh.co ...

  9. 前端编译原理 parser.js源码解读

    前面已经介绍了一个jison的使用,在正常开发中其实已经够用下,下面主要是看了下parser.js代码解读下,作为一些了解. 下面以最简单的文法产生的parser做一些代码注释 下面是一些注释,标示了 ...

随机推荐

  1. HDU 6068 Classic Quotation KMP+DP

    Classic Quotation Problem Description When online chatting, we can save what somebody said to form h ...

  2. socketIO原理图

  3. ie9浏览器中h标签的嵌套问题

    ie9中 h1-h6 标签中不能够嵌套h1-h6标签,否则往下看吧. 举个栗子: 我们要实现h1下的两个div实现左右分离,很简单吧? 看看html结构及css吧 <!DOCTYPE html& ...

  4. (linux)mmccard驱动的读写过程解析

      mmc io的读写从mmc_queue_thread()的获取queue里面的request开始. 先列出调用栈,看下大概的调用顺序, 下面的内容主要阐述这些函数如何工作. host->op ...

  5. linux设备驱动学习笔记(1)

    学习了将近半个月的设备驱动程序的编写,也有一些体会,这里写下来也给学习做一个总结,为后面的学习做更好的准备. 首先,个人感觉驱动程序的设计是很有套路的,最基本的要求就是要掌握这些套路.所谓的套路就是一 ...

  6. ES6 一些新特性的总结

    一.箭头函数 ES6中新增了一个箭头函数   ()=>,箭头函数通俗点讲就是匿名函数.箭头函数还有不同点在于改变函数中this,和js中的.bind  的方法差不多,继承后指向的不是最新的函数, ...

  7. codeforcfes Codeforces Round #287 (Div. 2) B. Amr and Pins

    B. Amr and Pins time limit per test 1 second memory limit per test 256 megabytes input standard inpu ...

  8. NOT IN clause and NULL values

    https://stackoverflow.com/questions/129077/not-in-clause-and-null-values This issue came up when I g ...

  9. YTU 2209: 建立链表(线性表)

    2209: 建立链表(线性表) 时间限制: 1 Sec  内存限制: 128 MB 提交: 282  解决: 185 题目描述 (线性表)设键盘输入n个英语单词,输入格式为n, w1, w2, -,w ...

  10. MYSQL进阶学习笔记四:MySQL存储过程之定义条件,处理过程及存储过程的管理!(视频序号:进阶_11,12)

    知识点五:MySQL存储过程之定义条件和处理过程及存储过程的管理(11,12) 定义条件和处理: 条件的定义和处理可以用来定义在处理过程中遇到的问题时相应的处理步骤. DECLARE CONTINUE ...