例如 通过学生获取学生所在学校信息,需要先查询学生所在班级,再通过班级查询所在学校信息。js代码类似写法如下:

function getStudentSchool(id) {
ajax.get("/Test/GetStudent", { "studentId": id }, function (student) {
if (student != null && student.ClassId != null) {
ajax.get("/Test/GetClass", { "classId": student.ClassId }, function (studentClass) {
if (studentClass != null && studentClass.SchoolId != null) {
ajax.get("/Test/GetSchool", { "schoolId": studentClass.SchoolId }, function (school) {
if (school != null) {
console.log(school);
}
});
}
});
}
});
} //调用入口方法
window.οnlοad= function(){
getStudentSchool(1);
};

  写了个类通过设置相关业务信号量来绑定触发的方法,当信号变量改变时就会自动调用相应的方法,改进方法如下:

function AsynFlag() {
if (typeof this.setFlag != "function") {
AsynFlag.prototype.setFlag = function (obj, name, fun) {
if (obj.hasOwnProperty(name)) {
obj[name + "_fun"] = fun;
return;
}
obj[name] = 0;
obj[name + "_"] = 0;
Object.defineProperty(obj, name, {
get: function () {
return obj[name + "_"];
},
set: function (value) {
if (value != obj[name + "_"]) {
obj[name + "_"] = value;
}
else
{
return;
}
if (obj[name + "_fun"] == null) {
obj[name + "_fun"] = fun;
}
obj[name + "_fun"]();
}
});
};
}
} var param = { "studentId": 0, "classId": 0, "schoolId": 0 };
var s = new AsynFlag();
var flag = {}; function getStudent()
{
ajax.get("/Test/GetStudent", { "studentId": param.studentId }, function (student) {
if (student != null && student.ClassId != null) {
param.classId = student.ClassId;
s.setFlag(flag, "canGetClass", getClass);
flag.canGetClass = true;
}
});
} function getClass()
{
ajax.get("/Test/GetClass", { "classId": param.ClassId }, function (studentClass) {
if (studentClass != null && studentClass.SchoolId != null) {
param.SchoolId = studentClass.SchoolId;
s.setFlag(flag, "canGetSchool", getSchool);
flag.canGetSchool = true;
}
});
} function getSchool()
{
ajax.get("/Test/GetSchool", { "schoolId": param.SchoolId }, function (school) {
if (school != null) {
console.log(school);
}
});
}
//调用入口方法
window.onload= function(){
param.studentId =1;
getStudent();
};

flag 是个信号量设置对象,s.setFlag(flag, "canGetClass", getClass); 设置flag拥有canGetClass属性,并且该属性绑定函数getClass,  当第一个ajax获得数据后设置信号并改变信号量触发绑定的getClass函数,flag对象中会自动创建canGetClass,

canGetClass_, 两个属性和一个canGetClass_fun方法来实现当canGetClass改变时调用canGetClass_fun=getClass。

以上代码实际测试可行。

代码已上传github https://github.com/SaFaJim/AsynFlag
————————————————
版权声明:本文为CSDN博主「皮皮虾大侠」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/Asa_Jim/article/details/86648199

js 回调地狱的另类解决方案尝试的更多相关文章

  1. JavaScript异步编程__“回调地狱”的一些解决方案

    异步编程在JavaScript中非常重要.过多的异步编程也带了回调嵌套的问题,本文会提供一些解决“回调地狱”的方法. setTimeout(function () { console.log('延时触 ...

  2. javascript回调地狱真的只能Promise来解决吗?js回调地狱,Promise。

    javascript的灵活在于函数可以当作函数的参数来传递,以及它的异步回调思想.但是这就带了一个很严重的问题,那就是回调次数过多,会影响代码结构,多层嵌套影响代码的可阅读性,也不便于书写. 举个例子 ...

  3. JS回调函数(深入篇)

    <有些错别字> 在Javascript中,函数是第一类对象,这意味着函数可以像对象一样按照第一类管理被使用.既然函数实际上是对象:它们能被“存储”在变量中,能作为函数参数被传递,能在函数中 ...

  4. JS回调函数深入篇

    <有些错别字> 在Javascript中,函数是第一类对象,这意味着函数可以像对象一样按照第一类管理被使用.既然函数实际上是对象:它们能被“存储”在变量中,能作为函数参数被传递,能在函数中 ...

  5. 【JavaScript】 使用Async 和 Promise 完美解决回调地狱

    很久以前就学习过Async和Promise,但总是一知半解的. 今天在写NodeJS的时候,发现好多第三方库使用回调,这样在实际操作中会出现多重回调,这就是传说中的JS回调地狱. 举个例子 有一个方法 ...

  6. 避免Node.js中回调地狱

    为了解决这个阻塞问题,JavaScript严重依赖于回调,这是在长时间运行的进程(IO,定时器等)完成后运行的函数,因此允许代码执行经过长时间运行的任务. downloadFile('example. ...

  7. js中的回调地狱 Callback to Hell

        本文重点:解决方式:1.promise  2. 拆解 function:将各步拆解为单个的 function  3. 通过 Generator 函数暂停执行的效果方式 4. 通过ES8的异步函 ...

  8. js中的回调函数 和promise解决异步操作中的回调地狱问题。

    回调函数 : 函数作为参数传递到另外一个函数中.简单数据类型和引入数据类型中的数组和对象作为参数传递大家肯定都不陌生,其实引用数据类型中的函数也是可以的. 事实上大家见到的很多,用到的也很多,比如jQ ...

  9. js动画实现&&回调地狱&&promise

    1. js实现动画 <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...

随机推荐

  1. 卸载&&更新docker(ubuntu)

    卸载docker: apt-get purge lxc-docker apt-get autoremove 更新docker: apt-get update apt-get install lxc-d ...

  2. openstack Mitaka实验环境安装(centos系统)

    本博文仅供参考,具体一定详细学习官方安装文档. 一 准备工作 二安装过程 1 安装NTP服务 2 安装openstack包 3 SQL数据库安装 4 安装消息队列 message queue 5 Me ...

  3. node.js+react全栈实践-Form中按照指定路径上传文件并

    书接上回,讲到“使用同一个新增弹框”中有未解决的问题,比如复杂的字段,文件,图片上传,这一篇就解决文件上传的问题.这里的场景是在新增弹出框中要上传一个图片,并且这个上传组件放在一个Form中,和其他文 ...

  4. [TimLinux] selinux sesearch命令详解

    1. 描述 sesearch用于搜索SELinux安全策略规则集,命令来自包:yum install setools-console. 2. 命令 命令使用方法: sesearch [OPTIONS] ...

  5. POJ 2186 Popular cows(SCC 缩点)

    Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <= 10, ...

  6. 洛谷 P2764(最小路径覆盖=节点数-最大匹配)

    给定有向图G=(V,E).设P 是G 的一个简单路(顶点不相交)的集合.如果V 中每个顶点恰好在P 的一条路上,则称P是G 的一个路径覆盖.P 中路径可以从V 的任何一个顶点开始,长度也是任意的,特别 ...

  7. HDU-1754 A - I Hate It

    很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老师有 ...

  8. 探究 Nginx 中 reload 流程的真相

    今天这篇文章主要来介绍下 Nginx 的 reload 流程.实际上在之前文章中,在更改了 nginx 配置文件时,我们都会执行 nginx -s reload 命令,我们执行这条命令的原因是希望 n ...

  9. Java构造器浅析

    Java构造器 问题引出 1.先看一段简单创建对象的例子: public class constructTest { public static void main(String[] args) { ...

  10. 基于 HTML5 WebGL 构建智能数字化城市 3D 全景

    前言 自 2011 年我国城镇化率首次突破 50% 以来,<新型城镇化发展规划>将智慧城市列为我国城市发展的三大目标之一,并提出到 2020 年,建成一批特色鲜明的智慧城市.截至现今,全国 ...