大厂面试题分享 面试题库
前后端面试题库 (面试必备) 推荐:★★★★★
地址:前端面试题库 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);
大厂面试题分享 面试题库
前后端面试题库 (面试必备) 推荐:★★★★★
地址:前端面试题库 web前端面试题库 VS java后端面试题库大全