json2.js 源码解读

这一部分是对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;
// }; 
时间: 2024-11-25 20:51:05

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

分享:json2.js源码解读笔记

1. 如何理解"json" 首先应该意识到,json是一种数据转换格式,既然是个"格式",就是个抽象的东西.它不是js对象,也不是字符串,它只是一种格式,一种规定而已. 这个格式规定了如何将js对象转换成字符串.以及转换成怎样的字符串--序列化 -- JSON.stringify 接口: 以及如何将一个有效字符串转换成js对象--反序列化-- JSON.parse 接口: 2. 关于作者 json作者是 道格拉斯.克劳福德 ,是一位js大牛,写过一本<java

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

1. 如何理解“json” 首先应该意识到,json是一种数据转换格式,既然是个“格式”,就是个抽象的东西.它不是js对象,也不是字符串,它只是一种格式,一种规定而已. 这个格式规定了如何将js对象转换成字符串.以及转换成怎样的字符串——序列化 —— JSON.stringify 接口: 以及如何将一个有效字符串转换成js对象——反序列化—— JSON.parse 接口: 2. 关于作者 json作者是 道格拉斯.克劳福德 ,是一位js大牛,写过一本<javascript语言精粹>,相信不少朋

json2.js源码解读记录

相关内容:json详细用法.js语法.unicode.正则 json特点--最简单.最小巧的经典js库. json作者:道克拉斯.克劳福德(Douglas Crockford)--js大牛 出版过一本书<js语言精粹> “ajax"中,a - 异步,j - js,(a - and), x - xml; xml -> json(异步没变.js没变) --说明json是一个可以取代xml的东西 json作为"格式",规定了: 什么样的字符串可以转换为js对象.如

fastclick.js源码解读分析

阅读优秀的js插件和库源码,可以加深我们对web开发的理解和提高js能力,本人能力有限,只能粗略读懂一些小型插件,这里带来对fastclick源码的解读,望各位大神不吝指教~! fastclick诞生背景与使用 在解读源码前,还是简单介绍下fastclick: 诞生背景 我们都知道,在移动端页面开发上,会出现一个问题,click事件会有300ms的延迟,这让用户感觉很不爽,感觉像是网页卡顿了一样,实际上,这是浏览器为了更好的判断用户的双击行为,移动浏览器都支持双击缩放或双击滚动的操作,比如一个链

prototype.js 源码解读(01)

prototype.js是一个设计的非常优雅且很有实用价值的js基础类库,其源码非常值得研究.研究它的源码不仅能提升个人水平,而且对你打下坚实的js基础也很有帮助.因本人技术水平有限,该解读仅供参考. 定义全局对象Prototype,包括属性版本号,属性版本号的定义有利于版本号的检测 9 var Prototype = { 10 Version: '1.5.0', 11 BrowserFeatures: { 12 XPath: !!document.evaluate 13 }, 14 15 Sc

prototype.js 源码解读(02)

如果你想研究一些比较大型的js框架的源码的话,本人建议你从其最初的版本开始研读,因为最初的版本东西少,易于研究,而后的版本基本都是在其基础上不断扩充罢了,所以,接下来我不准备完全解读prototype.js的源码了,而是拿它一些常见的API来解读. //定时器类,比起window.setInterval函数,该类能够使得回调函数不会被并发调用 var PeriodicalExecuter = Class.create();//Class类创建的定时器类 PeriodicalExecuter.pr

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

首先,我们先来简单说一下,require.js的原理: 1.载入模块? 2.通过模块名解析出模块信息,以及计算出URL? 3.通过创建SCRIPT的形式把模块加载到页面中.? 4.判断被加载的脚本,如果发现它有依赖就去加载依赖模块.如果不依赖其它模块,就直接执行factory方法 ?5.等所有脚本都被加载完毕就执行加载完成之后的回调函数. 从今天起,我们跟着我们简单的例子,通过跟踪代码,来了解require.js的源码. 1 <!DOCTYPE html> 2 <html lang=&q

Node.js源码解读-EventLoop

之前有一些机会断断续续接触过一些Node.js的项目,但只仅限于使用它简单实现一些功能,没有过多深入的研究.现在因为带领公司大前端技术,服务端渲染直出.BFF(Backend For Frontend)等需求会越来越多,组内需要对服务端技术有更深刻的理解,如果对Node.js仅仅停留在如何写业务代码的层面,那恐怕是无法保证以后服务的稳定性. 本文会基于node-v12.13.0版本的源码,对核心模块代码做一些阅读和理解,以窥探Node.js服务高效的秘诀.在研究源码之前,首先有几个疑问,看是否能

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

关于亚马逊的左侧菜单延迟,之前一直不知道它的实现原理.梦神提到了z三角,我也不知道这是什么东西.13号那天很有空,等领导们签字完我就可以走了.下午的时候,找到了一篇博客:http://jayuh.com/amazon-site-navigation/ 它提到亚马逊左侧菜单的秘密在于它有一个三角形的缓冲延迟区域. 当鼠标在这个蓝色范围内移动时,会有延迟,所以右侧的二级菜单才不会马上变化. 顺着博客在github找到了这个插件:https://github.com/jayuh/jQuery-menu