js开发代码片段与小技巧

简介: js开发代码片段与小技巧

1、+ 号的隐式类型转换使用
+[3] //3
+[1,2,3].slice(-1) //将 [3] 转换为了3
2、日期的转换
const addDaysToDate = (date, n) => {
const d = new Date(date);
d.setDate(d.getDate() + n);
return d.toISOString().split('T')[0];
};
addDaysToDate('2020-10-15', 10); // '2020-10-25'
平时做日期的计算可能更多的使用时间戳,秒数去加减,其实有对应的 setDate,getDate等方法去做(月份也一样)
toISOString 可以将日期对象转换为 YYYY-MM-DDTHH:mm:ss.sssZ 格式,分割掉 T 字符,即可以得到日期形式
3、对象的合并
Object.assign
// 在像某对象添加多个属性时,可以直接采取合并方式
const addStyles = (el, styles) => Object.assign(el.style, styles)
addStyles(document.getElementById('my-element'), {
background: 'red',
color: '#ffff00',
fontSize: '3rem'
})
4、Array.from
/
对于这个api知道更多的只是将类数组转化为真正的数组
(arrayLike,mapFn) 准确的说应该是类数组或者 可迭代对象({length: 5})
mapFn 接收一个回调,返回的数组每个元素会执行该回调
/
let a = Array.from({length: 10},(_,i) => i)
a // [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
5、关于工具函数封装tips
在封装工具类函数时,可以往闭包的角度靠(内部再返回一个函数)
有时候可以具有更好的拓展性
// 栗子:获取某些数中的最大值
const ary = (fn,n) => (...arg) => fn(...arg.slice(0,n))
const firstTwoMax = ary(Math.max, 2)
firstTwoMax(1,2,3,4)

// ary的函数将传入的函数,作为返回函数的返回值调用,类似于将传入的函数缓存下来,再一次调用时,传入比较值
// 这样有个好处是,其实我们可以更改比较条件,比如 取最小值等等
// 这样工具函数更具有灵活性
6、 按条件分割数组
const bifurcateBy = (arr, fn) =>
arr.reduce((acc, val, i) => (acc[fn(val, i) ? 0 : 1].push(val), acc), [
[],
[],
])
bifurcateBy(['beep', 'boop', 'foo', 'bar'], x => x[0] === 'b');
// [ ['beep', 'boop', 'bar'], ['foo'] ]

/
利用 reduce 将结果汇总,同时把分割条件作为一个 返回布尔值的函数传入
内部通过布尔值的判断,添加到对应的数组中
/
7、二分搜索算法,来查找已排序数组中某元素的位置
尽管现在有api可以直接实现位置查找,不过这个算法思想还是可以的
const binarySearch = (arr,item) => {
// 获取首位下标
let l = 0,
r = arr.length - 1;
while(l <= r) {
// 找到中间下标
const middle = Math.floor( (l + r) / 2)
const guess = arr[middle]
// 中间值与目标值比较
if(guess == item) return middle
// 中间值 > 目标值 ? 说明目标值在左侧,则取左边中间值再比较,反之亦然
guess > item ? r = middle - 1 : l = middle + 1
}
return -1
}
binarySearch([1,2,3,4],4) // 3
二分搜索算法,采取折中的方式,将中间值与目标做比较,然后不断修改中间值,进而找到目标值
8、冒泡排序
const bubbleSort = arr => {
let swapped = false;
const a = [...arr];
for (let i = 1; i < a.length; i++) {
swapped = false;
for (let j = 0; j < a.length - i; j++) {
if (a[j + 1] < a[j]) {
// es6 变量交换
[a[j], a[j + 1]] = [a[j + 1], a[j]];
swapped = true;
}
}
if (!swapped) return a;
}
return a;
};
核心思想是将当前值a 与 下一位置值b比较,如果 a > b,就把 a和b交互位置
同时,一轮排序其实就可以实现最大值排最后了,所以循环次数越往后会越少,这也是 i < a.length ,j < a.length - i 的原因
es6 的变量交换位置,更加方便了,其实也可以用于多个变量的赋值改变类似: [count, i] = [count + 1, r + 1]…
9、将字符串首字母转换为大写
// 有意思的是 这里只是通过解构的方式,就把字符串的首字符与剩余字符分开了
// 在此之前你说否还想着截取首字母再来转换?
const capitalize = ([first, ...rest], lowerRest = false) =>
first.toUpperCase() +
(lowerRest ? rest.join('').toLowerCase() : rest.join(''));

capitalize('fooBar', true); // 'Foobar'
10、链式控制异步流程
const chainAsync = fns => {
let curr = 0;
const last = fns[fns.length - 1];
// 内部实现一个函数,用于逐次从数组中提取执行函数
const next = () => {
const fn = fns[curr++];
// 提取函数与最后一个做对比,是否为最后一个函数
// 同时将这个内部函数,作为回调,传入下一个 异步函数中
fn === last ? fn() : fn(next);
};
next();
};

chainAsync([
next => {
console.log('0 seconds');
// 函数执行后,再执行传入的回调next
setTimeout(next, 1000);
},
next => {
console.log('1 second');
setTimeout(next, 1000);
},
() => {
console.log('2 second');
}
]);
异步的实现,其实还是回调的使用,只不过封装后,看似同步
好比 Promise的resolve,获取结果后,通过resolve回调出去
Promise + async ,运用迭代器,使得可以控制函数的暂停和恢复,更加具有同步风范
11、js 实现一个剪切板
const copyToClipboard = str => {
const el = document.createElement('textarea');
el.value = str;
el.setAttribute('readonly', '');
el.style.position = 'absolute';
el.style.left = '-9999px';
document.body.appendChild(el);
const selected =
document.getSelection().rangeCount > 0
? document.getSelection().getRangeAt(0)
: false;
el.select();
document.execCommand('copy');
document.body.removeChild(el);
if (selected) {
document.getSelection().removeAllRanges();
document.getSelection().addRange(selected);
}
}

copyToClipboard('Lorem ipsum')
12、根据给定函数或者属性,统计符合条件的数组元素的个数
const countBy = (arr, fn) =>
arr.map(typeof fn === 'function' ? fn : val => val[fn]).reduce((acc, val) => {
acc[val] = (acc[val] || 0) + 1;
return acc;
}, {});

countBy([6.1, 4.2, 6.3], Math.floor); // {4: 1, 6: 2}
countBy(['one', 'two', 'three'], 'length'); // {3: 2, 5: 1}
countBy([{ count: 5 }, { count: 10 }, { count: 5 }], x => x.count)
13、实现一个发布订阅模式
const createEventHub = () => ({
hub: Object.create(null),
emit(event, data) {
(this.hub[event] || []).forEach(handler => handler(data));
},
on(event, handler) {
if (!this.hub[event]) this.hub[event] = [];
this.hub[event].push(handler);
},
off(event, handler) {
const i = (this.hub[event] || []).findIndex(h => h === handler);
if (i > -1) this.hub[event].splice(i, 1);
if (this.hub[event].length === 0) delete this.hub[event];
}
});
//代码效果参考:http://www.zidongmutanji.com/zsjx/306135.html

const handler = data => console.log(data);
const hub = createEventHub();
let increment = 0;

// Subscribe:
hub.on('message', handler);
hub.on('message', () => console.log('Message event fired'));
hub.on('increment', () => increment++);

// Publish
hub.emit('message', 'hello world');
hub.emit('message', { hello: 'world' });
hub.emit('increment');

// Unsubscribe
hub.off('message', handler);
14、补充一点关于函数length属性
/*

  • js 函数的length是表示参数的个数,需要注意的是
  • 该个数指的是 函数必须接收参数的个数,且是有效参数(应当是默认值前的参数个数)
  • */
    function a( a,b = 10,c) {} //a.lenght = 1
      a(10,'',30)        // a = 10 b = '' c = 30
      a(10,null,30)      // a = 10 b = null c = 30
      a(10,undefiend,30) // a =10 b = 10 c = 30
    
    // 对于包含默认值的参数,只有传递 undefiend 时(或者不传)才是取默认值,其他情况均为传啥就是啥
相关文章
|
2月前
|
Web App开发 JavaScript 前端开发
Node.js 是一种基于 Chrome V8 引擎的后端开发技术,以其高效、灵活著称。本文将介绍 Node.js 的基础概念
Node.js 是一种基于 Chrome V8 引擎的后端开发技术,以其高效、灵活著称。本文将介绍 Node.js 的基础概念,包括事件驱动、单线程模型和模块系统;探讨其安装配置、核心模块使用、实战应用如搭建 Web 服务器、文件操作及实时通信;分析项目结构与开发流程,讨论其优势与挑战,并通过案例展示 Node.js 在实际项目中的应用,旨在帮助开发者更好地掌握这一强大工具。
61 1
|
1月前
|
Web App开发 JavaScript 前端开发
Node.js开发
Node.js开发
57 13
|
2月前
|
存储 JavaScript 前端开发
深入浅出Node.js后端开发
在数字化时代的浪潮中,后端开发作为连接用户与数据的桥梁,扮演着至关重要的角色。本文将以Node.js为例,深入探讨其背后的哲学思想、核心特性以及在实际项目中的应用,旨在为读者揭示Node.js如何优雅地处理高并发请求,并通过实践案例加深理解。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的视角和思考。
|
2月前
|
Web App开发 开发框架 JavaScript
深入浅出Node.js后端开发
本文将带你领略Node.js的魅力,从基础概念到实践应用,一步步深入理解并掌握Node.js在后端开发中的运用。我们将通过实例学习如何搭建一个基本的Web服务,探讨Node.js的事件驱动和非阻塞I/O模型,以及如何利用其强大的生态系统进行高效的后端开发。无论你是前端开发者还是后端新手,这篇文章都会为你打开一扇通往全栈开发的大门。
|
2月前
|
Web App开发 开发框架 JavaScript
深入浅出Node.js后端开发
在这篇文章中,我们将一起探索Node.js的奇妙世界。无论你是刚接触后端开发的新手,还是希望深化理解的老手,这篇文章都适合你。我们将从基础概念开始,逐步深入到实际应用,最后通过一个代码示例来巩固所学知识。让我们一起开启这段旅程吧!
|
1月前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
本文将带领读者从零基础开始,一步步深入到Node.js后端开发的精髓。我们将通过通俗易懂的语言和实际代码示例,探索Node.js的强大功能及其在现代Web开发中的应用。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的见解和技巧,让你的后端开发技能更上一层楼。
|
2月前
|
JavaScript 前端开发 API
深入理解Node.js事件循环及其在后端开发中的应用
本文旨在揭示Node.js的核心特性之一——事件循环,并探讨其对后端开发实践的深远影响。通过剖析事件循环的工作原理和关键组件,我们不仅能够更好地理解Node.js的非阻塞I/O模型,还能学会如何优化我们的后端应用以提高性能和响应能力。文章将结合实例分析事件循环在处理大量并发请求时的优势,以及如何避免常见的编程陷阱,从而为读者提供从理论到实践的全面指导。
|
2月前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
本文将带你走进Node.js的世界,从基础到进阶,逐步解析Node.js在后端开发中的应用。我们将通过实例来理解Node.js的异步特性、事件驱动模型以及如何利用它处理高并发请求。此外,文章还会介绍如何搭建一个基本的Node.js服务器,并探讨如何利用现代前端框架与Node.js进行交互,实现全栈式开发。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的视角和深入的理解。
48 4
|
2月前
|
前端开发 JavaScript 关系型数据库
基于 Vue2.0 + Nest.js 全栈开发的后台应用
Vue2 Admin 是一个基于 Vue2 和 Ant Design Pro 开发的前端项目,配合 Nest.js 构建的后端,提供了一个完整的全栈后台应用解决方案。该项目支持动态国际化、用户权限管理、操作日志记录等功能,适合全栈开发者学习参考。线上预览地址:https://vue2.baiwumm.com/,用户名:Admin,密码:abc123456。
|
2月前
|
开发框架 JavaScript 前端开发
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势。通过明确的类型定义,TypeScript 能够在编码阶段发现潜在错误,提高代码质量;支持组件的清晰定义与复用,增强代码的可维护性;与 React、Vue 等框架结合,提供更佳的开发体验;适用于大型项目,优化代码结构和性能。随着 Web 技术的发展,TypeScript 的应用前景广阔,将继续引领 Web 开发的新趋势。
53 2