聊聊从Vue.js源码中我学到了精妙方法

开发 前端
如果我们的应用需要创建很多新的对象,并且这些对象还有许多的方法,为了节省内存,我们建议把这些方法都定义在构造函数的 prototype 属性上。当然,在某些情况下,我们需要将某些方法定义在构造函数中,这种情况一般是因为我们需要访问构造函数内部的私有变量。

[[393248]]

话不多说,赶快试试尤大大教给我们的这几个精妙方法吧!在工作中肯定会用得到。

立即执行函数

页面加载完成后只执行一次的设置函数。

  1. (function (a, b) { 
  2.   console.log(a, b); // 1,2 
  3. })(1, 2); 

通常,全局变量被作为一个参数传递给立即执行参数,这样它在函数内部不使用window也可以被访问到。

  1. (function (global) { 
  2.   console.log(global); // Window对象 
  3. })(this); 

多层嵌套三目运算符

三目运算符嵌套的写法,使得代码可读性差,简单业务场景下可以试着使用。

  1. var a = 1; 
  2. var b = 0; 
  3. a == 1 ? (b == 2 ? (b = 3) : (b = 1)) : ""
  4. console.log(b); // 1 

冻结对象

不可对指定对象增删改。

  1. var emptyObject = Object.freeze({ 
  2.   key"1"
  3. }); 
  4. emptyObject.name = "maomin"
  5. delete emptyObject.key
  6. emptyObject.key = "2"
  7. console.log(emptyObject); 

密封对象

只能对指定对象进行更改,不可进行增加删除操作。

  1. var sealObject = Object.seal({ 
  2.   key: 3, 
  3. }); 
  4. sealObject.name = "maomin"
  5. delete sealObject.key
  6. sealObject.key = 4; 
  7. console.log(sealObject); // 4 

检查是否是原始值

  1. function isPrimitive(value) { 
  2.   return ( 
  3.     typeof value === "string" || 
  4.     typeof value === "number" || 
  5.     // $flow-disable-line 
  6.     typeof value === "symbol" || 
  7.     typeof value === "boolean" 
  8.   ); 

快速检测是否是对象

当我们知道原始值时,它主要用于将对象与原始值区分开来。

  1. function isObject(obj) { 
  2.   return obj !== null && typeof obj === "object"
  3. console.log(isObject(true)); //false 

检测目标类型

  1. var _toString = Object.prototype.toString; 
  2.  
  3. function toRawType(value) { 
  4.   return _toString.call(value).slice(8, -1); 
  5. console.log(toRawType([])); // Array 

检查目标是否是有效的数组索引

  1. function isValidArrayIndex(val) { 
  2.   var n = parseFloat(String(val)); 
  3.   return n >= 0 && Math.floor(n) === n && isFinite(val); 

检测是否是Promise对象

  1. function isDef(v) { 
  2.   return v !== undefined && v !== null
  3. function isPromise(val) { 
  4.   return ( 
  5.     isDef(val) && 
  6.     typeof val.then === "function" && 
  7.     typeof val.catch === "function" 
  8.   ); 
  9. var promiseObj = new Promise(function (resolve, reject) { 
  10.   // 一段耗时的异步操作 
  11.   resolve("成功"); // 数据处理完成 
  12.   // reject('失败') // 数据处理出错 
  13. }).then
  14.   (res) => { 
  15.     console.log(res); 
  16.   }, // 成功 
  17.   (err) => { 
  18.     console.log(err); 
  19.   } // 失败 
  20. ); 
  21. console.log(isPromise(promiseObj)); // true 

目标转换为字符串

  1. var _toString = Object.prototype.toString; 
  2. function isPlainObject(obj) { 
  3.   return _toString.call(obj) === "[object Object]"
  4. function toString(val) { 
  5.   return val == null 
  6.     ? "" 
  7.     : Array.isArray(val) || (isPlainObject(val) && val.toString === _toString) 
  8.     ? JSON.stringify(val, null, 2) 
  9.     : String(val); 
  10. console.log(toString({ name: 1 })); // {"name": 1} 

转化为数字

将输入值转换为数字以便持久化。如果转换失败,则返回原始字符串。

  1. function toNumber(val) { 
  2.   var n = parseFloat(val); 
  3.   return isNaN(n) ? val : n; 

检测key是否在创建的Map对象内

  1. function makeMap(str, expectsLowerCase) { 
  2.   var map = Object.create(null); 
  3.   var list = str.split(","); 
  4.   for (var i = 0; i < list.length; i++) { 
  5.     map[list[i]] = true
  6.   } 
  7.   return expectsLowerCase 
  8.     ? function (val) { 
  9.         return map[val.toLowerCase()]; 
  10.       } 
  11.     : function (val) { 
  12.         return map[val]; 
  13.       }; 
  14. var isBuiltInTag = makeMap("slot,component"true); 
  15. console.log(isBuiltInTag("component")); // true 

删除简单数组中某一项

  1. function remove(arr, item) { 
  2.   if (arr.length) { 
  3.     var index = arr.indexOf(item); 
  4.     if (index > -1) { 
  5.       return arr.splice(index, 1); 
  6.     } 
  7.   } 
  8. console.log(remove([1, 2], 1)); // [1] 

检测对象中是否有指定key

  1. var hasOwnProperty = Object.prototype.hasOwnProperty; 
  2. function hasOwn(obj, key) { 
  3.   return hasOwnProperty.call(obj, key); 
  4. console.log(hasOwn({ name: 1 }, "name")); //true 

将类数组对象转化为真实数组

  1. function toArray(list, start) { 
  2.   start = start || 0; 
  3.   var i = list.length - start; 
  4.   var ret = new Array(i); 
  5.   while (i--) { 
  6.     ret[i] = list[i + start]; 
  7.   } 
  8.   return ret; 
  9. console.log(toArray({ 0: 42, 1: 52, 2: 63, length: 3 })); // [42, 52, 63] 

将属性混合到目标对象中

  1. function extend(to, _from) { 
  2.   for (var key in _from) { 
  3.     to[key] = _from[key]; 
  4.   } 
  5.   return to
  6. console.log(extend({ name: 1 }, { name1: 2 })); // {name:1,name1:2} 

将对象数组合并为单个对象

  1. function extend(to, _from) { 
  2.   for (var key in _from) { 
  3.     to[key] = _from[key]; 
  4.   } 
  5.   return to
  6. function toObject(arr) { 
  7.   var res = {}; 
  8.   for (var i = 0; i < arr.length; i++) { 
  9.     if (arr[i]) { 
  10.       extend(res, arr[i]); 
  11.     } 
  12.   } 
  13.   return res; 
  14. console.log(toObject([{ name: 1 }, { name: 1 }, { name: 2 }, { name1: 3 }])); // {name: 2, name1: 3} 

检测指定项在数组(简单数组、数组对象)中的索引

  1. function isObject(obj) { 
  2.   return obj !== null && typeof obj === "object"
  3. function looseEqual(a, b) { 
  4.   if (a === b) { 
  5.     return true
  6.   } 
  7.   var isObjectA = isObject(a); 
  8.   var isObjectB = isObject(b); 
  9.   if (isObjectA && isObjectB) { 
  10.     try { 
  11.       var isArrayA = Array.isArray(a); 
  12.       var isArrayB = Array.isArray(b); 
  13.       if (isArrayA && isArrayB) { 
  14.         return ( 
  15.           a.length === b.length && 
  16.           a.every(function (e, i) { 
  17.             return looseEqual(e, b[i]); 
  18.           }) 
  19.         ); 
  20.       } else if (a instanceof Date && b instanceof Date) { 
  21.         return a.getTime() === b.getTime(); 
  22.       } else if (!isArrayA && !isArrayB) { 
  23.         var keysA = Object.keys(a); 
  24.         var keysB = Object.keys(b); 
  25.         return ( 
  26.           keysA.length === keysB.length && 
  27.           keysA.every(function (key) { 
  28.             return looseEqual(a[key], b[key]); 
  29.           }) 
  30.         ); 
  31.       } else { 
  32.         /* istanbul ignore next */ 
  33.         return false
  34.       } 
  35.     } catch (e) { 
  36.       /* istanbul ignore next */ 
  37.       return false
  38.     } 
  39.   } else if (!isObjectA && !isObjectB) { 
  40.     return String(a) === String(b); 
  41.   } else { 
  42.     return false
  43.   } 
  44.  
  45. function looseIndexOf(arr, val) { 
  46.   for (var i = 0; i < arr.length; i++) { 
  47.     if (looseEqual(arr[i], val)) { 
  48.       return i; 
  49.     } 
  50.   } 
  51.   return -1; 
  52. console.log(looseIndexOf([{ name: 1 }, { name: 2 }], 4)); // -1 
  53. console.log(looseIndexOf([{ name: 1 }, { name: 2 }], { name: 1 })); // 0 

确保函数只调用一次

  1. function once(fn) { 
  2.   var called = false
  3.   return function () { 
  4.     if (!called) { 
  5.       called = true
  6.       fn.apply(this, arguments); 
  7.     } 
  8.   }; 
  9. var callOnce = once(function () { 
  10.   console.log("javascript"); 
  11. }); 
  12. callOnce(); // javascript 
  13. callOnce(); 

定义对象属性

如果你想禁止一个对象添加新属性并且保留已有属性,就可以使用Object.preventExtensions(obj)。

  1. function def(obj, key, val, enumerable) { 
  2.   Object.defineProperty(obj, key, { 
  3.     value: val, // 对象定义属性 
  4.     enumerable: !!enumerable, // 描述属性是否会出现在for in 或者 Object.keys()的遍历中 
  5.     writable: true, // 是否可写 
  6.     configurable: true, // 是否重新定义或者删除 
  7.   }); 
  8. var obj = { 
  9.   name: 1, 
  10. }; 
  11. def(obj, "name1", 2, true); 
  12. obj.name1 = 3; 
  13. console.log(obj); // {name: 1, name1: 3} 

浏览器环境嗅探

  1. var inBrowser = typeof window !== "undefined"
  2. var inWeex = typeof WXEnvironment !== "undefined" && !!WXEnvironment.platform; 
  3. var weexPlatform = inWeex && WXEnvironment.platform.toLowerCase(); 
  4. var UA = inBrowser && window.navigator.userAgent.toLowerCase(); 
  5.  
  6. var isIE = UA && /msie|trident/.test(UA); 
  7. var isIE9 = UA && UA.indexOf("msie 9.0") > 0; 
  8. var isEdge = UA && UA.indexOf("edge/") > 0; 
  9. var isAndroid = (UA && UA.indexOf("android") > 0) || weexPlatform === "android"
  10. var isIOS = (UA && /iphone|ipad|ipod|ios/.test(UA)) || weexPlatform === "ios"
  11. var isChrome = UA && /chrome\/\d+/.test(UA) && !isEdge; 
  12. var isPhantomJS = UA && /phantomjs/.test(UA); 
  13. var isFF = UA && UA.match(/firefox\/(\d+)/); 

JS构造函数内的方法与构造函数prototype属性上方法的对比

定义在构造函数内部的方法,会在它的每一个实例上都克隆这个方法;定义在构造函数的 prototype 属性上的方法会让它的所有示例都共享这个方法,但是不会在每个实例的内部重新定义这个方法。如果我们的应用需要创建很多新的对象,并且这些对象还有许多的方法,为了节省内存,我们建议把这些方法都定义在构造函数的 prototype 属性上。当然,在某些情况下,我们需要将某些方法定义在构造函数中,这种情况一般是因为我们需要访问构造函数内部的私有变量。

  1. function A() { 
  2.   this.say = function () { 
  3.     console.log(1); 
  4.   }; 
  5. var a = new A(); 
  6. a.say(); 
  7. function B() {} 
  8. B.prototype.say = function () { 
  9.   console.log(2); 
  10. }; 
  11. var b = new B(); 
  12. b.say(); 
  13. var c = new B(); 
  14. c.say(); 

获取标签内容(包含标签)

  1. function getOuterHTML(el) { 
  2.   if (el.outerHTML) { 
  3.     return el.outerHTML; 
  4.   } else { 
  5.     var container = document.createElement("div"); 
  6.     container.appendChild(el.cloneNode(true)); 
  7.     return container.innerHTML; 
  8.   } 

字符串hash值

  1. function hash(str) { 
  2.   var hash = 5381; 
  3.   var i = str.length; 
  4.   while (i) { 
  5.     hash = (hash * 33) ^ str.charCodeAt(--i); 
  6.   } 
  7.   return hash >>> 0; 
  8. console.log(hash("222sd"));// 164533792 

 

责任编辑:武晓燕 来源: maomin9761
相关推荐

2021-01-02 09:48:13

函数运算js

2022-03-27 09:06:04

React类型定义前端

2021-03-09 09:55:02

Vuejs前端代码

2024-05-13 08:04:26

Vue.jsWeb应用程序

2021-05-08 06:14:28

Vue.js片段开发

2016-01-18 10:06:05

编程

2020-02-22 15:01:51

后端前端开发

2020-04-01 08:40:44

Vue.jsweb开发

2016-09-21 13:32:13

JavascriptWeb前端

2021-03-13 11:23:51

源码逻辑框架

2020-09-25 06:32:25

前端

2020-12-31 10:47:03

开发Vuejs技术

2021-10-25 05:43:40

前端技术编程

2020-06-01 14:02:25

Vue.js框架模板

2016-09-21 12:56:31

JavascriptWeb前端

2020-10-27 08:07:17

Vue.js

2020-06-14 15:09:00

JavaScript开发技术

2023-04-26 22:52:19

视觉人脸检测人脸对齐

2024-04-12 08:54:13

从库数据库应用

2022-12-01 17:13:44

点赞
收藏

51CTO技术栈公众号