大厂面试题分享 面试题库
前后端面试题库 (面试必备) 推荐:★★★★★
地址:前端面试题库 web前端面试题库 VS java后端面试题库大全
1.图片失败,重新加载
如果图片资源不存在,那可以设置图片失败的占位图片。但是如果图片资源是存在的,只不过因为网络慢(资源在国外),而导致图片加载失败的,应该是让图片重新加载,而不是失败一次就直接设置失败的占位图片。
所以监听图片加载失败的时候,让图片重新加载。
原理是:重新给图片src赋值时,会触发加载资源的动作,去重新加载图片资源。此时可以监听全局的加载失败事件,然后重新给图片的src赋值资源
// 监听全局失败事件 window.addEventListener( 'error', e => { // 如果不是图片,则退出执行 if (e.target.nodeName !== 'IMG') return /** 获取图片重新请求次数的属性 */ let retry = Number(e.target.getAttribute('retry')) // 开启一个定时器,这里每400ms执行一次 const timer = setTimeout(() => { // 如果重试次数大于2,则取消重试,设置图片错误占位图片 if (retry > 2) { // 图片使用占位图片url e.target.src = '' // 清除定时器 clearTimeout(timer) } else { // 次数+1 retry += 1 // 设置重试属性 e.target.setAttribute('retry', retry) // 重新获取图片 e.target.src = e.target.src } }, 400) }, true ) 复制代码
把这段代码粘贴到油猴脚本,然后到github测试效果:
2.循环执行异步操作后执行
在执行异步任务的时候,都会返回一个可以停止事件的函数,调用函数就可以停止任务。任务的多少是取决于实际操作的,而不是你决定的。你需要将这些任务都关闭后,再去做其他操作。
首先创建一个queue数组作为store来储存这些函数,每次执行的时候都把返回的函数储存到数组里面,然后再执行。
const queue = [fn1, fn2, ...] async function start(){ // 等待所有任务都完成 await Promise.all( queue.map(async (callback)=>{ await callback() }) ) // 去做其他的事情 do_otherthing() } 复制代码
3.异步队列执行
有时候频繁接收一些任务,但是还没来得及执行,所以需要把这些任务都放到队列里面,然后再按顺序执行。这时候就可以写一个数组来存储这些任务。
const queue = [fn1, fn2, ...] async function start() { if (queuq.length == 0) return await queue[0]() queue.shift() start() } //或者使用for循环解决 for(let i=0;i<queue.length;i++){ await queue[i]() } 复制代码
2和3有什么差异呢?此时我们可以模拟异步请求测试一下
function sleep1() { return new Promise(r => { setTimeout(() => { console.log(1) r(0) }, 5000) }) } function sleep2() { return new Promise(r => { setTimeout(() => { console.log(2) r(0) }, 5000) }) } const queue = [sleep1, sleep2] async function start1(){ // 等待所有任务都完成 await Promise.all( queue.map(async (callback)=>{ await callback() }) ) // 去做其他的事情 console.log("start1任务完成") } async function start2() { if (queue.length == 0) return console.log("start2任务完成") await queue[0]() queue.shift() start2() } start1() start2() 复制代码
对于2,是5秒后全部打印结果,也就是全都是异步操作,互不干扰的。
而对于3,是5秒后打印一次,然后又隔5秒又打印一次,是同步执行的。
4.刷新自动滚动到顶部
参考张鑫旭大佬的教程:www.bilibili.com/video/BV15U…
刷新浏览器的时候,浏览器都会记住浏览的位置。但有时候我们就不想要记住位置,而是刷新就自动回到顶部了,此时可以:
if(history.scrollRestoration){ history.scrollRestoration = "manual" } 复制代码
5.a标签下载文件
下载文件社区有一个包file-saver,但实际上也是利用了a标签下载,利用简单的几行代码就可以使用了,没必要特地引入一个包。
// 获取到的数据 const data = xxx // 创建a标签下载 const a = document.createElement('a'); // 存到二进制大对象中 const blob = new Blob([data]); const url = URL.createObjectURL(blob); a.setAttribute('href', url); a.setAttribute('download', `测试.txt`); a.click(); URL.revokeObjectURL(url);