js异步状态监控
说明:写这篇文章,是希望被吐槽的。
一、背景
在做报表页面的时候,页面上有很多的异步加载,而设计的loading是个全局的,一个页面就有一个。
控制loading什么时候出现,什么时候消失,要实时的知道页面上异步加载的东西是否执行完毕,只有所有的异步都加载完,loading才能停止。
并且,如果用户操作了页面,某个局部又要开始加载,loading要被通知到,执行loading效果。
问题的难点有两个:
1.怎么知道所有的异步都加载完闭了呢?
2.如何通知loading?
二、思路
1.思路一
设计一个外部状态map,异步执行完毕,就修改外部变量的值。开一个计时器,不断的这个map,如果全部加载完毕,就停止loading。
// 状态集合
var statusMap = {
"异步1": false,
"异步2": false
// ...
};
// 计时器不断检查状态
setInterval(function () {
// 检查statusMap
if (isReady(statusMap)) {
// show loading
} else {
// hide loading
}
}, 1000);
//异步
$.ajax({
//...
success: function () {
statusMap["异步1"] = true;
}
});
这个方案虽然能解决问题,但是太不够优雅,呵呵呵呵......
2.思路二
使用promise或者jQuery的deferred(@上位者的怜悯 提供的建议)。
$.when(异步1,异步2).done(function(){
// show loading
}).fail(function(){
// hide loading
});
这个方案有个弊端,一旦异步加载完毕,不能修改promise的状态了,因此不适合这个场景。
3.思路三
自己造轮子基于事件监听的,一个状态改变,就扫一遍所有监听的对象,最后再根据用状态,回调对应的函数
三、思路三实现
(function (win) {
var watch = function () {
var that = this;
// arr
if (arguments.length == 1 && Object.prototype.toString.call(arguments[0]) === "[object Array]") {
that.watchArr = arguments[0];
} else {
that.watchArr = Array.prototype.slice.call(arguments);
}
that.watchArr.forEach(function (fairy) {
fairy.watch = that;
});
that.isReady = true;
return that;
};
watch.prototype = {
ready: function (callback) {
this.readyHandler = callback;
this.check();
return this;
},
unready: function (callback) {
this.unReadyHandler = callback;
this.check();
return this;
},
add: function (fairy, check) {
this.watchArr.push(fairy);
if (check === undefined || check === true) {
this.check();
}
return this;
},
check: function () {
var nowIsReady = this.watchArr.every(function (fairy) {
return fairy.isReady;
});
if (nowIsReady != this.isReady) {
if (nowIsReady) {
if (this.readyHandler) {
this.readyHandler.call();
this.isReady = true;
}
} else {
if (this.unReadyHandler) {
this.unReadyHandler.call();
this.isReady = false;
}
}
}
},
setIsReadyById: function (id, isReady, check) {
var index = this.watchArr.indexOf(function (fairy) {
fairy.id = id;
});
this.watchArr[index].isReady = isReady;
if (check === undefined || check === true) {
this.check();
}
},
setIsReadyAll: function (isReady, check) {
this.watchArr.forEach(function (fairy) {
fairy.isReady = isReady;
});
if (check === undefined || check === true) {
this.check();
}
}
};
var fairy = function (isReady) {
this.id = new Date().getTime();
this.isReady = isReady || false;
};
fairy.prototype = {
toReady: function () {
this.isReady = true;
this.watch.check();
},
toUnReady: function () {
this.isReady = false;
this.watch.check();
}
};
win.Watch = watch;
win.Fairy = fairy;
})(window);
// 用例
var loader = new Loader();
var listFairy = new Fairy();
var chartFairy = new Fairy();
var watch = new Watch(listFairy, chartFairy);
watch.ready(function () {
loader.stop();
}).unready(function () {
loader.start();
});
//异步
$.ajax({
//...
success: function () {
listFairy.toReady();
}
});
纳尼,说好的事件呢,哪里去了。
好吧,吐槽点来了,实现的过程中,发现其实没必要用事件。
实现到最后,发现它其实就是一个变相的思路一。
the end 未完不续 ...
js异步状态监控的更多相关文章
- JS魔法堂:深究JS异步编程模型
前言 上周5在公司作了关于JS异步编程模型的技术分享,可能是内容太干的缘故吧,最后从大家的表情看出"这条粉肠到底在说啥?"的结果:(下面是PPT的讲义,具体的PPT和示例代码在h ...
- js异步编程
前言 以一个煮饭的例子开始,例如有三件事,A是买菜.B是买肉.C是洗米,最终的结果是为了煮一餐饭.为了最后一餐饭,可以三件事一起做,也可以轮流做,也可能C需要最后做(等A.B做完),这三件事是相关的, ...
- 深究JS异步编程模型
前言 上周5在公司作了关于JS异步编程模型的技术分享,可能是内容太干的缘故吧,最后从大家的表情看出"这条粉肠到底在说啥?"的结果:(下面是PPT的讲义,具体的PPT和示例代码在h ...
- JS异步加载的三种方式
js加载的缺点:加载工具方法没必要阻塞文档,过得js加载会影响页面效率,一旦网速不好,那么整个网站将等待js加载而不进行后续渲染等工作. 有些工具方法需要按需加载,用到再加载,不用不加载,. 默认正常 ...
- 转:web前端面试题合集 (Javascript相关)(js异步加载详解)
1. HTTP协议的状态消息都有哪些? 1**:请求收到,继续处理2**:操作成功收到,分析.接受3**:完成此请求必须进一步处理4**:请求包含一个错误语法或不能完成5**:服务器执行一个完全有效请 ...
- JS异步加载的三种方案
js加载的缺点:加载工具方法没必要阻塞文档,个别js加载会影响页面效率,一旦网速不好,那么整个网站将等待js加载而不进行后续渲染等工作. 有些工具方法需要按需加载,用到再加载,不用不加载. 一.def ...
- JS 异步系列 —— Promise 札记
Promise 研究 Promise 的动机大体有以下几点: 对其 api 的不熟悉以及对实现机制的好奇; 很多库(比如 fetch)是基于 Promise 封装的,那么要了解这些库的前置条件得先熟悉 ...
- 深入理解node.js异步编程:基础篇
###[本文是基础内容,大神请绕道,才疏学浅,难免纰漏,请各位轻喷] ##1. 概述 目前开源社区最火热的技术当属Node.js莫属了,作为使用Javascript为主要开发语言的服务器端编程技术和平 ...
- 前端分享----JS异步编程+ES6箭头函数
前端分享----JS异步编程+ES6箭头函数 ##概述Javascript语言的执行环境是"单线程"(single thread).所谓"单线程",就是指一次只 ...
随机推荐
- mysql自动备份维护shell脚本 (copy)
#!/bin/bash #Mysql 自动备份 压缩并上传到 指定ftp #设想每天凌晨3点备份mysql #编辑crontab配置文件 # * * * backupmysql.sh #压缩并以&qu ...
- 烂泥:vcenter5.5无AD下的安装与配置
本文由ilanniweb提供友情赞助,首发于烂泥行天下 想要获得更多的文章,可以关注我的微信ilanniweb. 公司现在的虚拟化使用的基本上都是vsphere,目前大约有7台物理机,为了更好的管理虚 ...
- dos2unix用法
windows下编写的脚本,上传到linux上执行,有时会出现莫名其妙的结果,也是折腾了半天才找出原因,windows和linux有些编码不同,经dos2unix处理后,结果一切正常man dos2u ...
- vim linux SecureCRT vim颜色的设置
解决方法: 1.首先按照环境一的方法检查一下,如果环境一的情况不在进行第二步. 2.修改终端TERM这个环境变量: 1)export命令 临时生效 # echo $TERM xterm #如果结果 ...
- IE自动跳转到标准模式
<meta http-equiv="X-UA-Compatible" content="IE=edge">
- 2016"百度之星" - 初赛(Astar Round2B)
Problem Description 中位数定义为所有值从小到大排序后排在正中间的那个数,如果值有偶数个,通常取最中间的两个数值的平均数作为中位数. 现在有n个数,每个数都是独一无二的,求出每个数在 ...
- BZOJ1216[HNOI2003]操作系统 [模拟 优选队列]
1216: [HNOI2003]操作系统 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 754 Solved: 421[Submit][Status ...
- AC日记——单词倒排 1.7 28
28:单词倒排 总时间限制: 1000ms 内存限制: 65536kB 描述 编写程序,读入一行英文(只包含字母和空格,单词间以单个空格分隔),将所有单词的顺序倒排并输出,依然以单个空格分隔. 输 ...
- 重写成员“log4net.Util.ReadOnlyPropertiesDictionary.GetObjectData(System.Runtime.Serialization.SerializationInfo, System.Runtime.Serialization.StreamingContext)”时违反了继承安全性规则
在.NET 4.0下使用最新版本的log4Net 1.2.10,会遇到下面这样的错误: 重写成员“log4net.Util.ReadOnlyPropertiesDictionary.GetObject ...
- [No000022]他们说:得诺贝尔奖到底有多难?