A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

本帖最后由 没名字i 于 2018-8-9 12:44 编辑

描述
由于JS里for循环是同步任务,而reader.onloadend是异步任务,所以同步的会先执行完循环结束,等到异步回调引用外部变量所以,始终是循环最后一次值,so,归纳了下面三种方式

Promise
Promise.all函数可以并行调用参数中的Promise对象方法,并且将所有Promise对象方法返回值作为数组输入到then回调中

fileReaderPromise = (file) =>{
   let promise = new Promise((resolve, reject) =>{
            let reader = new FileReader()
            reader.readAsDataURL(file)
            reader.onloadend = (e) => {
                let new_result = e.target.result
                resolve(new_result)
            }
   });
   return promise;
}

handle_images = (files) => {
    let filesPromise = files.map(file => this.fileReaderPromise(file))
    Promise.all(filesPromise)
        .then(data => {
            data.map((image,index) => console.log('image',image,'index',index))
        })
}
自执行函数
创建自执行函数局部作用域,保证外部变量不受影响

handle_images = (files) => {
    _.map(files, (file,index) => {
        (function(new_file,new_index){
            let reader = new FileReader()
            reader.readAsDataURL(file)
            reader.onloadend = (e) => {
                let new_result = e.target.result
                //...这里你可以引用new_file,new_index
                console.log('new_file',new_file,'new_index',new_index)
            }
         }.bind(this)(file,index)
    }
}

递归
handle_images = (i) => {
    let new_i = i
    if (new_i<files.length){
        let file = files
        let reader = new FileReader()
        reader.readAsDataURL(file)
        reader.onloadend = (e) => {
            let new_result = e.target.result
            //...
            console.log('file',file,'i',i)
            
            //逻辑处理完后
            new_i++;
            this.handle_images(new_i)
        }
    }
}
//调用
this.handle_images(0)

转载:https://juejin.im/entry/5b6b9ee2 ... d_browser_extension

1 个回复

倒序浏览
奈斯
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马