当前位置:网站首页>Share 14 JS functions you must know
Share 14 JS functions you must know
2022-08-03 17:39:00 【Front-end talent】
英文 | https://javascript.plainenglish.io/you-must-understand-these-14-javasript-functions-1f4fa1c620e2
翻译 | 杨小爱
1、Determine the concrete type of any object
众所周知,JavaScript There are six primitive data types(Boolean、Number、String、Null、Undefined、Symbol)and an object data type.But did you know that object data types can be subdivided into many seed types?An object may be an array、函数、map等,If we want to get the concrete type of the object,应该怎么做呢?
代码:
function toRawType (value) {
let _toString = Object.prototype.toString;
let str = _toString.call(value)
return str.slice(8, -1)
}
解释
ECMAScript 有以下规则:
对于不同的对象,调用 Object.prototype.toString() returns different results.
而且,Object.prototype.toString() 的返回值总是‘[object’+‘tag’+‘]’的格式.If we only want the middle label,We can pass regex or String.prototype.slice()Delete characters on both sides.
例子:
toRawType(null)
// "Null"
toRawType(/sdfsd/)
//"RegExp"
2、Cache function calculation results
如果有这样的功能:
function computed(str) {
// Suppose the calculation in the funtion is very time consuming
console.log('2000s have passed')
return 'a result'
}
We want to cache the result of the function operation, when called later,如果参数相同,The function is no longer executed,而是直接返回缓存中的结果.我们能做什么?
代码:
function cached(fn){
// Create an object to store the results returned after each function execution.
const cache = Object.create(null);
// Returns the wrapped function
return function cachedFn (str) {
// If the cache is not hit, the function will be executed
if ( !cache[str] ) {
let result = fn(str);
// Store the result of the function execution in the cache
cache[str] = result;
}
return cache[str]
}
}
例子:
3、实现Array.prototype.map
这是 JavaScript A useful built-in method in ,You should be able to implement this functionality yourself.
代码:
const selfMap = function (fn, context) {
let arr = Array.prototype.slice.call(this)
let mappedArr = Array()
for (let i = 0; i < arr.length; i++) {
if (!arr.hasOwnProperty(i)) continue;
mappedArr[i] = fn.call(context, arr[i], i, this)
}
return mappedArr
}
Array.prototype.selfMap = selfMap;
例子:
4、实现Array.prototype.filter
这是 JavaScript A useful built-in method in ,You should be able to implement this functionality yourself.
代码:
const selfFilter = function (fn, context) {
let arr = Array.prototype.slice.call(this)
let filteredArr = []
for (let i = 0; i < arr.length; i++) {
if(!arr.hasOwnProperty(i)) continue;
fn.call(context, arr[i], i, this) && filteredArr.push(arr[i])
}
return filteredArr
}
Array.prototype.selfFilter = selfFilter;
例子:
5、实现 Array.prototype.some
这是 JavaScript A useful built-in method in ,You should be able to implement this functionality yourself.
代码:
const selfSome = function (fn, context) {
let arr = Array.prototype.slice.call(this)
if(!arr.length) return false
for (let i = 0; i < arr.length; i++) {
if(!arr.hasOwnProperty(i)) continue;
let res = fn.call(context,arr[i],i,this)
if(res)return true
}
return false
}
Array.prototype.selfSome = selfSome;
例子:
6、实现 Array.prototype.reduce
这是 JavaScript A useful built-in method in ,You should be able to implement this functionality yourself.
代码:
const selfReduce = function (fn, initialValue) {
let arr = Array.prototype.slice.call(this)
let res
let startIndex
if (initialValue === undefined) {
for (let i = 0; i < arr.length; i++) {
if (!arr.hasOwnProperty(i)) continue
startIndex = i
res = arr[i]
break
}
} else {
res = initialValue
}
for (let i = ++startIndex || 0; i < arr.length; i++) {
if (!arr.hasOwnProperty(i)) continue
res = fn.call(null, res, arr[i], i, this)
}
return res
}
Array.prototype.selfReduce = selfReduce;
例子:
7、实现 Array.prototype.flat
代码:
const selfFlat = function (depth = 1) {
let arr = Array.prototype.slice.call(this)
if (depth === 0) return arr
return arr.reduce((pre, cur) => {
if (Array.isArray(cur)) {
return [...pre, ...selfFlat.call(cur, depth - 1)]
} else {
return [...pre, cur]
}
}, [])
}
Array.prototype.selfFlat = selfFlat;
例子:
8、柯里化
Currying is a technique for evaluating a function with multiple arguments as a sequence of functions with a single argument.
换句话说,When a function does not accept all arguments at once,Instead it takes the first argument and returns a new function,The function accepts a second parameter and returns a new function,The function accepts a third parameter,依此类推,until all parameters have been fulfilled.
That's where we call the function add(1,2,3) 转换为 add(1)(2)(3) .通过使用这种技术,Small blocks can be easily configured and reused.
为什么有用?
Currying helps you avoid passing the same variable over and over again.
It helps to create higher order functions,It is very helpful for event handling.
Widgets can be easily configured and reused.
Let's look at a simple add function.It accepts three operands as parameters,and returns the sum of all three operands as the result.
function add(a,b,c){
return a + b + c;
}
You can use too little(结果奇怪)或太多(多余的参数被忽略)来调用它.
add(1,2,3) --> 6
add(1,2) --> NaN
add(1,2,3,4) --> 6 //Extra parameters will be ignored.
How to convert an existing function to curried 版本?
代码:
function curry(fn) {
if (fn.length <= 1) return fn;
const generator = (...args) => {
if (fn.length === args.length) {
return fn(...args)
} else {
return (...args2) => {
return generator(...args, ...args2)
}
}
}
return generator
}
例子:
9、去抖动
Debounce is nothing more than reducing unnecessary time-consuming computations,以提高浏览器性能.在某些情况下,Some functions require more time to perform an action.例如,Take, for example, a search bar on an e-commerce site.
Suppose the user wants to get“Tutorix 学习套件”.He types every character of the product into the search bar.After entering each character,It happens once from browser to server Api 调用,to get the desired product.Because he wants to“Tutorix 学习套件”,The user has to do it from the browser to the server 17 次 Api 调用.
想象一个场景,Billions are called when millions of people do the same search Api 时.So billions of calls at a time Api It will definitely lead to slower browser performance.In order to reduce this disadvantage,Debounce comes in.
在这种情况下,Debounce will set a time interval between keystrokes,假设为 2 秒.If the time between keystrokes exceeds 2 秒,则只会进行 Api 调用.在这 2 秒内,The user can enter at least some characters,从而减少 Api call these characters.由于 Api 调用减少,Browser performance will improve.必须注意,It is updated with every keystroke Debouncing 功能.
代码:
const debounce = (func, time = 17, options = {
leading: true,
context: null
}) => {
let timer;
const _debounce = function (...args) {
if (timer) {
clearTimeout(timer)
}
if (options.leading && !timer) {
timer = setTimeout(null, time)
func.apply(options.context, args)
}else{
timer = setTimeout(() => {
func.apply(options.context, args)
timer = null
}, time)
}
};
_debounce.cancel = function () {
clearTimeout(timer)
timer = null
};
return _debounce
};
10、 节流
Throttling will change the function in such a way,i.e. it can fire at most once in an interval.例如,No matter how many times the user clicks the button,The limit will be at 1000 The function is executed only once in milliseconds.
代码:
const throttle = (func, time = 17, options = {
leading: true,
trailing: false,
context: null
}) => {
let previous = new Date(0).getTime()
let timer;
const _throttle = function (...args) {
let now = new Date().getTime();
if (!options.leading) {
if (timer) return
timer = setTimeout(() => {
timer = null
func.apply(options.context, args)
}, time)
} else if (now - previous > time) {
func.apply(options.context, args)
previous = now
} else if (options.trailing) {
clearTimeout(timer)
timer = setTimeout(() => {
func.apply(options.context, args)
}, time)
}
};
_throttle.cancel = () => {
previous = 0;
clearTimeout(timer);
timer = null
};
return _throttle
};
11、 延迟加载图片
Lazy loading images means loading images asynchronously on a website——也就是说,After the above-the-fold content is fully loaded,even conditionally,Only if they appear in the browser's viewport.
This means if the user doesn't scroll all the way down,then images placed at the bottom of the page won't even be loaded.
代码:
// getBoundingClientRect
let imgList1 = [...document.querySelectorAll(".get_bounding_rect")]
let num = imgList1.length
let lazyLoad1 = (function () {
let count = 0
return function () {
let deleteIndexList = []
imgList1.forEach((img,index) => {
let rect = img.getBoundingClientRect()
if (rect.top < window.innerHeight) {
img.src = img.dataset.src
// Add picture to delete list after loading successfully
deleteIndexList.push(index)
count++
if (count === num) {
//When all pictures are loaded, unbind scroll event
document.removeEventListener('scroll',lazyLoad1)
}
}
})
// Delete loaded pictures
imgList1 = imgList1.filter((_,index)=>!deleteIndexList.includes(index))
}
})()
12、The array is randomly unordered
We often need to shuffle an array.
代码:
// Randomly select one of all elements after the current element to exchange with the current element
function shuffle(arr) {
for (let i = 0; i < arr.length; i++) {
let randomIndex = i + Math.floor(Math.random() * (arr.length - i));
[arr[i], arr[randomIndex]] = [arr[randomIndex], arr[i]]
}
return arr
}
// Generate a new array, randomly take an element from the original array and put it into the new array
function shuffle2(arr) {
let _arr = []
while (arr.length) {
let randomIndex = Math.floor(Math.random() * (arr.length))
_arr.push(arr.splice(randomIndex, 1)[0])
}
return _arr
}
例子:
13、单例模式
The singleton pattern limits the number of instances of a particular object to one,This single instance is called the singleton pattern.
Singletons are useful in situations where system-wide operations need to be coordinated from a single central location.An example is database connection pooling.The pool manages the creation of all database connections for the entire application、Destruction and Lifecycle,确保没有连接“丢失”.
Singletons reduce the need for global variables,这在 JavaScript 中尤为重要,Because it limits namespace pollution and the associated risk of name collisions.
代码:
function proxy(func) {
let instance;
let handler = {
construct(target, args) {
if (!instance) {
// Create an instance if there is not exist
instance = Reflect.construct(func,args)
}
return instance
}
}
return new Proxy(func, handler)
}
// example
function Person(name, age) {
this.name = name
this.age = age
}
const SingletonPerson = proxy(Person)
let person1 = new SingletonPerson('zhl', 22)
let person2 = new SingletonPerson('cyw', 22)
console.log(person1 === person2) // true
例子:
14、实现 JSON.stringify
这是 JavaScript A useful built-in method in ,You should be able to implement this functionality yourself.
代码:
const isString = value => typeof value === 'string';
const isSymbol = value => typeof value === 'symbol'
const isUndefined = value => typeof value === 'undefined'
const isDate = obj => Object.prototype.toString.call(obj) === '[object Date]'
const isFunction = obj => Object.prototype.toString.call(obj) === '[object Function]';
const isComplexDataType = value => (typeof value === 'object' || typeof value === 'function') && value !== null;
const isValidBasicDataType = value => value !== undefined && !isSymbol(value);
const isValidObj = obj => Array.isArray(obj) || Object.prototype.toString.call(obj) === '[object Object]';
const isInfinity = value => value === Infinity || value === -Infinity
// Symbol,undefined,function in array will become null
// Infinity,NaN will also become null
const processSpecialValueInArray = value =>
isSymbol(value) || isFunction(value) || isUndefined(value) || isInfinity(value) || isNaN(value) ? null : value;
// Handling property values according to JSON specification
const processValue = value => {
if (isInfinity(value) || isNaN(value)) {
return null
}
if (isString(value)) {
return `"${value}"`
}
return value
};
// obj.loop = obj
const jsonStringify = (function () {
// Closure + WeakMap prevent circular references
let wp = new WeakMap();
//It is the function in the closure that recursively calls jsonstrify, not the jsonstrify function declared by const
return function jsonStringify(obj) {
if (wp.get(obj)) throw new TypeError('Converting circular structure to JSON');
let res = "";
if (isComplexDataType(obj)) {
if (obj.toJSON) return obj.toJSON;
if (!isValidObj(obj)) {
return
}
wp.set(obj, obj);
if (Array.isArray(obj)) {
res += "[";
let temp = [];
obj.forEach((value) => {
temp.push(
isComplexDataType(value) && !isFunction(value) ?
jsonStringify(value) :
`${processSpecialValueInArray(value, true)}`
)
});
res += `${temp.join(',')}]`
} else {
res += "{";
let temp = [];
Object.keys(obj).forEach((key) => {
if (isComplexDataType(obj[key])) {
if (isValidObj(obj[key])) {
temp.push(`"${key}":${jsonStringify(obj[key])}`)
} else if (isDate(obj[key])) {
temp.push(`"${key}":"${obj[key].toISOString()}"`)
} else if (!isFunction(obj[key])) {
temp.push(`"${key}":{}`)
}
} else if (isValidBasicDataType(obj[key])) {
temp.push(`"${key}":${processValue(obj[key])}`)
}
});
res += `${temp.join(',')}}`
}
} else if (isSymbol(obj)) {
return
} else {
return obj
}
return res
}
})();
// example
let s = Symbol('s')
let obj = {
str: "123",
arr: [1, {e: 1}, s, () => {
}, undefined,Infinity,NaN],
obj: {a: 1},
Infinity: -Infinity,
nan: NaN,
undef: undefined,
symbol: s,
date: new Date(),
reg: /123/g,
func: () => {
},
dom: document.querySelector('body'),
};
console.log(jsonStringify(obj));
console.log(JSON.stringify(obj));
例子:
总结
That's all I share with you14个JavaScript的函数,These functions are also us as onewebA must-know for front-end developers,希望对你有用,如果觉得对你有帮助的话,请记得点赞我,关注我,并将它分享给你身边做开发的朋友,也许能够帮助到他.
学习更多技能
请点击下方公众号
边栏推荐
猜你喜欢
随机推荐
一加Ace值得买吗?用实力诠释性能的强大
FinClip | July 2022 Product Highlights
如何避免无效的沟通
论文解读(JKnet)《Representation Learning on Graphs with Jumping Knowledge Networks》
cell delay and net delay
图像传感第一章学习心得
opencv 直方图比较
websocket Handshake failed due to invalid Upgrade header
C# 构造函数如人之影子
PMP试题 | 每日一练,快速提分
mysql之的执行计划
【机器学习】机器学习基本概念/术语3
uniapp 切换 history 路由模
深度学习跟踪DLT (deep learning tracker)
国内首发可视化智能调优平台,小龙带你玩转KeenTune UI
Web3的开源为何会如此受到人们喜爱?
面试突击:什么是粘包和半包?怎么解决?
three.js简介
Interpretation of the paper (JKnet) "Representation Learning on Graphs with Jumping Knowledge Networks"
CAD如何自定义快捷键