摘要:如果方法的第二個參數(shù)存在,返回的調(diào)用結(jié)果,否則直接返回編譯結(jié)果。
這一部分是對Date String Number Boolean擴(kuò)展toString方法,Date的toString是返回UTC格式的字符串,而后面幾個是返回原始值。
function f(n) {// 返回兩位數(shù)字字符串 return n < 10 ? ‘0‘ + n: n; } if (typeof Date.prototype.toJSON !== ‘function‘) {//如果Date不支持原生的toJSON方法 Date.prototype.toJSON = function() {//擴(kuò)展Date的toJSON方法 //是否是有窮數(shù),如果為true,返回根據(jù)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; }; //擴(kuò)展String Number Boolean的toJSON方法 String.prototype.toJSON = Number.prototype.toJSON = Boolean.prototype.toJSON = function() { return this.valueOf();//返回他們的原始值 }; }
這里是擴(kuò)展Stringify方法
function str(key, holder) {//第一次調(diào)用時 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];//第二次調(diào)用時 value就是傳入的鍵所對應(yīng)的值 //如果value有toJSON方法 if (value && typeof value === ‘object‘ && typeof value.toJSON === ‘function‘) { value = value.toJSON(key);//調(diào)用value.toJSON方法 } if (typeof rep === ‘function‘) {//如果replace是一個方法 value = rep.call(holder, key, value); } // 判斷value類型 switch (typeof value) { case ‘string‘://如果是字符串,加引號 return quote(value); case ‘number‘://如果是數(shù)值 //有窮數(shù)用原生的String()將數(shù)值轉(zhuǎn)為符串,否則返回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 = [];//臨時數(shù)組 if (Object.prototype.toString.apply(value) === ‘[object Array]‘) {//數(shù)組 length = value.length; for (i = 0; i < length; i += 1) {//對數(shù)組的每一項遞歸調(diào)用str partial[i] = str(i, value) || ‘null‘; } // 如果partial為[],返回"[]" // 如果 gap分隔符存在,返回[ ‘ + gap + partial.join(‘, ‘ + gap) + ‘ ‘ + mind + ‘]‘ // 如果分隔符不存在,返回‘[‘ + partial.join(‘,‘) + ‘]‘ v = partial.length === 0 ? ‘[]‘: gap ? ‘[ ‘ + gap + partial.join(‘, ‘ + gap) + ‘ ‘ + mind + ‘]‘: ‘[‘ + partial.join(‘,‘) + ‘]‘; gap = mind;//重置為"" return v; } if (rep && typeof rep === ‘object‘) {//如果rep存在且為數(shù)組或者對象 length = rep.length;//如果是數(shù)組 for (i = 0; i < length; i += 1) {//過濾 if (typeof rep[i] === ‘string‘) { k = rep[i];//鍵是數(shù)組的值 v = str(k, value);//遞歸調(diào)用 if (v) { //"key": value 或者"key":value partial.push(quote(k) + (gap ? ‘: ‘: ‘:‘) + v); } } } } else {//如果不是數(shù)組或方法或不存在 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 ? ‘{ ‘ + gap + partial.join(‘, ‘ + gap) + ‘ ‘ + mind + ‘}‘: ‘{‘ + partial.join(‘,‘) + ‘}‘; gap = mind; return v; } }
下面是圖解:
function quote(string) {//將傳入的字符串加上引號,有必要轉(zhuǎn)義的先轉(zhuǎn)義 escapable.lastIndex = 0;//起始位置從0開始 return escapable.test(string) ? ‘"‘ + string.replace(escapable, function(a) {//匹配到的字符 如 var c = meta[a];//字符對應(yīng)的轉(zhuǎn)義表示 //如果c是字符串 ,直接返回對象中鍵所對應(yīng)的值 ‘ ‘=>‘ ‘ //如果c不是字符串,也就是說它不在meta對象中,這時做不同的轉(zhuǎn)義處理: //拿 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 + ‘"‘; }
這里是擴(kuò)展parse方法
if (typeof JSON.parse !== ‘function‘) {//如果JSON沒有parse方法 cx = /[u0000u00adu0600-u0604u070fu17b4u17b5u200c-u200fu2028-u202fu2060-u206fufeffufff0-uffff]/g; JSON.parse = function(text, reviver) {//擴(kuò)展JSON方法 var j; function walk(holder, key) {// walk({‘‘:j},‘‘) var k, v, value = holder[key];//value第一次是傳入的eval編譯后的結(jié)果 if (value && typeof value === ‘object‘) {//如果value存在并且是對象 for (k in value) { if (Object.prototype.hasOwnProperty.call(value, k)) {//是否有原型上的屬性 v = walk(value, k);//遞歸調(diào)用獲取結(jié)果 if (v !== undefined) { value[k] = v; } else { delete value[k]; } } } } //調(diào)用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 因為需要轉(zhuǎn)義 return ‘u‘ + (‘0000‘ + a.charCodeAt(0).toString(16)).slice( - 4); }); } // text.replace(/(?:["/bfnrt]|u[0-9a-fA-F]{4})/g, ‘@‘) // =>把 uffff 這類轉(zhuǎn)為@ // =>var text=‘{"a":" 44","b":"uffff"}‘; text.replace(/(?:["/bfnrt]|u[0-9a-fA-F]{4})/g, ‘@‘); //{"a":"@44","b":"@"} // replace(/"[^" ]*"|true|false|null|-?d+(?:.d*)?(?:[eE][+-]?d+)?/g, ‘]‘) // =>把非空字符串、數(shù)值、bool、null替換為] // .replace(/(?:^|:|,)(?:s*[)+/g, ‘‘) // => 把 [ ,[ :[ 這類的替換為‘‘ // 如果剩余的字符串只剩下 ]:,{}和空格 就是測試通過,否則拋出異常 if (/^[],:{}s]*$/.test(text.replace(/(?:["/bfnrt]|u[0-9a-fA-F]{4})/g, ‘@‘) .replace(/"[^" ]*"|true|false|null|-?d+(?:.d*)?(?:[eE][+-]?d+)?/g, ‘]‘).replace(/(?:^|:|,)(?:s*[)+/g, ‘‘))) { j = eval(‘(‘ + text + ‘)‘);//eval()傳入的參數(shù)加括號編譯 return typeof reviver === ‘function‘ ? walk({//如果第二個參數(shù)是函數(shù) 返回walk({‘‘:j},‘‘)的結(jié)果,否則直接返回eval編譯的結(jié)果 ‘‘: j }, ‘‘) : j; } throw new SyntaxError(‘JSON.parse‘); }; }
通過一系列的替換操作,如果剩下的字符串只剩下 ]:,{}和空格,測試通過,接下來就可以用eval編譯。
如果parse方法的第二個參數(shù)存在,返回walk的調(diào)用結(jié)果,否則直接返回eval編譯結(jié)果。
//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; // };
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/82965.html
摘要:字符串轉(zhuǎn)對象代表字符串對象轉(zhuǎn)字符串代表對象運(yùn)用時候需要除了以外需要包切記哦 字符串轉(zhuǎn)對象(strJSON代表json字符串) var obj = eval(strJSON); var obj = strJSON.parseJSON(); var obj = JSON.parse(strJSON); json對象轉(zhuǎn)字符串(obj代表json對象) var str = o...
摘要:個人博客已上線,歡迎前去訪問評論無媛無故的個人博客以下內(nèi)容若有問題煩請即時告知我予以修改,以免誤導(dǎo)更多人。把一個邏輯值轉(zhuǎn)換為字符串,并返回結(jié)果。注冊后,可直接調(diào)用的接口,并獲取的返回值。 個人博客已上線,歡迎前去訪問評論!無媛無故 - wangchloe的個人博客 以下內(nèi)容若有問題煩請即時告知我予以修改,以免誤導(dǎo)更多人。 toString() && valueOf() JSON ...
摘要:個人博客已上線,歡迎前去訪問評論無媛無故的個人博客以下內(nèi)容若有問題煩請即時告知我予以修改,以免誤導(dǎo)更多人。把一個邏輯值轉(zhuǎn)換為字符串,并返回結(jié)果。注冊后,可直接調(diào)用的接口,并獲取的返回值。 個人博客已上線,歡迎前去訪問評論!無媛無故 - wangchloe的個人博客 以下內(nèi)容若有問題煩請即時告知我予以修改,以免誤導(dǎo)更多人。 toString() && valueOf() JSON ...
摘要:個人博客已上線,歡迎前去訪問評論無媛無故的個人博客以下內(nèi)容若有問題煩請即時告知我予以修改,以免誤導(dǎo)更多人。把一個邏輯值轉(zhuǎn)換為字符串,并返回結(jié)果。注冊后,可直接調(diào)用的接口,并獲取的返回值。 個人博客已上線,歡迎前去訪問評論!無媛無故 - wangchloe的個人博客 以下內(nèi)容若有問題煩請即時告知我予以修改,以免誤導(dǎo)更多人。 toString() && valueOf() JSON ...
閱讀 3771·2023-04-25 20:00
閱讀 3118·2021-09-22 15:09
閱讀 513·2021-08-25 09:40
閱讀 3424·2021-07-26 23:38
閱讀 2213·2019-08-30 15:53
閱讀 1101·2019-08-30 13:46
閱讀 2799·2019-08-29 16:44
閱讀 2052·2019-08-29 15:32