位置: 编程技术 - 正文

有关Promises异步问题详解(异步promise原理)

编辑:rootadmin

推荐整理分享有关Promises异步问题详解(异步promise原理),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:promise实现异步的原理,promise解决异步的原理和方法,异步请求方式及promise用法,promisify一个异步函数,异步请求方式及promise用法,promise异步函数,promise实现异步的原理,promisify一个异步函数,内容如对您有帮助,希望把文章链接给更多的朋友!

迄今为止,可能每个JavaScript开发者和他们的祖母都听说过Promises。如果你没有,那么你即将会。promises的概念是由CommonJS小组的成员在 Promises/A规范 中提出来的。Promises被逐渐用作一种管理异步操作回调的方法,但出于它们的设计,它们远比那个有用得多。事实上,由于它们的多种用法,有无数人告诉我——在我写过一些关于promises的东西后——我“遗漏了promises的重点”。那么什么是promises的重点呢?

一点关于Promises的东西

在我开始promise的“重点”之前,我想我应该给你一点它们如何工作的内貌。一个promise是一个对象——根据Promise/A规范——只需要一个方法:then。then方法带有三个参数:一个成功回调,一个失败回调,和一个前进回调(规范没有要求包括前进回调的实现,但是很多都实现了)。一个全新的promise对象从每个then的调用中返回。一个promise可以是三种状态之一:未完成的,完成的,或者失败的。promise以未完成的状态开始,如果成功它将会是完成态,如果失败将会是失败态。当一个promise移动到完成态,所有注册到它的成功回调将被调用,而且会将成功的结果值传给它。另外,任何注册到promise的成功回调,将会在它已经完成以后立即被调用。

同样的事情发生在promise移动到失败态的时候,除了它调用的是失败回调而不是成功回调。对包含前进特性的实现来说,promise在它离开未完成状态以前的任何时刻,都可以更新它的progress。当progress被更新,所有的前进回调(progress callbacks)会被传递以progress的值,并被立即调用。前进回调被以不同于成功和失败回调的方式处理;如果你在一个progress更新已经发生以后注册了一个前进回调,新的前进回调只会在它被注册以后被已更新的progress调用。我们不会进一步深入promise状态是如何管理的,因为那不在规范之内,而且每个实现都有差别。在后面的例子中,你将会看到它是如何完成的,但目前这就是所有你需要知道的。

处理回调

像前面提到的为异步操作处理回调,是promises的最基本和最普通的用途,让我们将一个标准的回调与一个采用了promise的回调比较一下。

我很怀疑只是看到这个例子的话是否有人会真的关心去使用promises。看起来没有什么好处,除了“then”使得在异步操作完成之后的回调函数被调用这件事看起来更加明显。但是即使是这个好处,我们现在有了更多的代码(抽象应该使我们的代码更短,不是吗?)而且promise比标准回调稍微性能差一点。

但是,不要让这阻碍到你。如果这就是promise可以做的最好的事,这篇文章就不会存在了

厄运的金字塔

正如你所见,promises的使用使得事情变扁平而且更可读了。这能起作用是因为——像早先提到的——then返回了一个promise,所以你可以将then的调用不停的串连起来。由then返回的promise装载了由调用返回的值。如果调用返回了一个promise(像这个例子中的情形一样),then返回的 promise装载了与你的回调返回的promise所装载的相同值。这内容很多,因此我将帮助你一步一步的理解它

异步操作返回一个promise对象。因此我们在那个promise对象中调用then,并且传给它一个回调函数;then也会返回一个promise。当异步操作结束,它将给promise装上数据。然后(第一次)回调被调用,数据被作为参数传递进去。如果回调不含有返回值,then返回的promise将会立即不带值组装。如果回调返回的不是一个promise,那么then返回的 promise将会立即装载那个数值。如果回调返回一个promise(像例子中的),那么then返回的 promise将等待直到我们回调返回的promise被完全装载。一旦我们回调的 promise被装载,它装载的值(本例中就是data2)将会被提交给then的promise。然后then中的promise装载了data2。等等。听起来有点复杂,但事实上很简单,如果我说的你不能理解,我非常抱歉。我猜我可能不是谈论它的最佳人选。

用命名的回调替代

但显然 promises 不是使这个结构扁平化的唯一方法。在写了一篇提到promises解决了厄运的金字塔问题的帖子之后,有个人对该帖评论说……

我想promises有时是有用的,但是“嵌套”的回调的问题(圣诞树综合症)可以仅用一个命名的函数作为一个参数替代匿名函数的方法平常的处理:

它的例子只是给出了一层深的例子,但它仍是正确的。我们来扩展我前面的例子,使这个看起来容易些。

命名回调

看看上面的代码!他们绝对是对的!它就是一个扁平的结构,但是这里有个问题同样也存在于 我以前从来没有注意过的老的回调例子中:依赖性和复用性。依赖性和复用性是相互关联的可逆类型。一样东西依赖的越少,那么它的复用性就越大。在以上的例子中,handler1依赖handler2,handler2依赖handler3.这就意味着handler1无论出于任何目的都不可在被用除非handler2也呈现出来。假如你不打算重用他们,那么给你的函数命名又有什么意义呢?

最糟糕的的是handler1都不关心在handler2里面发生了什么事情。它压根就不需要handler2除了和它异步工作。因此,让我们消除这些依赖性然后通过用promise使函数更具复用性。

链式回调

有关Promises异步问题详解(异步promise原理)

这样看起来是不是好多了?假如另外的函数存在的话,现在handler1和handler2都互不相关了。想看看他们是否真的很棒呢?现在handler1可以被用在不需要handler2的情况下了。相反,handler1被操作以后,我们将可以用另一个handler。

复用函数

现在handler1已经从handler2脱离而且可以被用在了更多的情形中,特别是那些由handler2提供的功能而我们又不想用的。这就是复用性!评论家解决代码易读性的唯一方法就是通过消除缩进。我们不想消除缩进仅仅是为了缩进。多层次的缩进仅仅是某些事情错误的标志,问题不一定在它本身。他就像是由脱水引起的头痛。真正的问题是脱水,不是头痛。解决的方法是获得水合物,而不是用一些止痛药。

并行异步操作

在前面我提到的文章里,我将promises与events在处理异步操作方面做了比较。遗憾的是,按照那些曾提到过的人在评论里给的说法,我比较的不是很成功。我描述出了promises的力量,接着转到events来描述它们的力量,就像在我的特别项目里用到的那样。没有比较和对比。一位评论者写道(修改了一点语法错误):

我想用帖子中的例子是一个坏的对照。有篇论文证明了promises的值将会怎样,如果按下虚构的“启动服务器按钮”,将不仅仅是启动一个web服务器,还有一个数据库服务器,当它们都在运行的时候只是更新了UI。

使用promise的.when方法将会使这种“多个异步操作”例子变得普通,然而响应多个异步事件需要一个并不普通的代码量。他完全正确。事实上我没有比较那两种情况。那篇文章的要点实际在于说明promises不是异步操作的唯一机制,而且在一些情况下,它们也不一定是最好的。在这个评论者指出的情况下,promises当然是最佳的解决办法。我们来看看他说的是什么

jQuery 具有 一个名为when的方法 ,可以带上任意数量的promise参数,并返回一个单一的promise。如果任何一个promise传入失败,when返回的promise也会失败。如果所有的promises被装载,那么每个值都将会按照promises被定义的顺序传递给附加的回调。

以并行的方式执行无数的异步操作非常有用,然后只要在它们之中的每一个结束之后继续执行回调。我们看一个简单的例子。

jQuery.when

人们经常说这是 promises 带来的最好的东西之一,也是 promises 的一部分重要的意义所在。我也认为这是个简化了大量操作的好特性,但是这种 when 方法的机制 根本就没有在任何 Promises 规范中提到,所以我不认为它是 Promises意义所在。有一个规范提到了 when 方法,但是和上面的完全不同。就我所知,jQuery 是唯一的实现了这种 when 方法的库。其他的 promises 库,例如 Q, Dojo, 和 when 依照 Promises/B spec 实现了 when 方法, 但是并没有实现注释者提及的 when 方法。但是,Q 库有一个 all方法,when.js 也有一个 parallel方法,与上面的 jQuery.when 方法作用一样,只是它们接受一个数组类型的参数,而不是任意数量的参数。

值的表示

Promise是处理以下场景的更好的方法:

"我想在这个数据库中找一个用户,但find方法是异步的。"

因此,这里我们有了一个不能立刻返回值的find方法。但最终它确实"返回"了一个数值(通过一个回调的方式),而你希望以某种方式处理那个数值。现在,通过使用一个回调,你能定义一个继续部分,或者说“一些将在以后时间里处理那个数值的代码”

Promise改变了那种“嘿,这里是一些你会发现你用来处理返回数值的代码”。它们是一些允许"find"方法说“嘿,我将忙着找你要找的信息,但与此同时你能继续等着返回结果,而且你能同时以任何你希望的方式处理它,就像实际的东西!”

Promise代表了真实的数值。那就是陷阱。它们工作在你像处理实际东西一样处理Promise的时候。Promise的JavaScript实现期待你给它传递一个回调函数,这只是一个“巧合”,它不是重要的事情。

我相信这真的就是promise的重点。为什么?读一读 Promise/A规范 的第一句“一个promise代表了一个操作的一次完成最终返回的数值。“使它有点明显了,是不是?好吧,即使那就是重点,那也不能阻止我在后面本文中呈现其他人的见解。不管怎么说,我们再多谈论这个思想一点。

结论

promise的重点是它代表一个操作返回的最终结果值,但使用它们的原因是使同步操作更好的并行。自从异步编程进入此场景,到处都是弹出的回调,以奇怪的方式遮住我们的代码。Promise是一种改变其的方法。Promise允许我们以同步的方式写代码,同时给予我们代码的异步执行。

每天一篇javascript学习小结(Date对象) 1、Date.now()//Date.now()isinECMAScript5//Priortothat,use+newDate()//获取当前时间varnow=(typeofDate.now=="function"Date.now():+newDate());alert("Rightnow:"+now);2、Date.parse()方法varnow=new

javascript:void(0)是什么意思及href=#与href=javascriptvoid(0)的区别 Javascript中void是一个操作符,该操作符指定要计算一个表达式但是不返回值。void操作符用法格式如下:1.javascript:void(expression)2.javascript:voidexpressionexpressio

javascript:void(0)点击登录没反应怎么解决 巧用批处理解决IE不支持JavaScript等问题rem=====批处理开始========regsvractxprxy.dllregsvrshdocvw.dllRegsvrURLMON.DLLRegsvractxprxy.dllRegsvrshdocvw.dllregsvroleaut.dllru

标签: 异步promise原理

本文链接地址:https://www.jiuchutong.com/biancheng/384591.html 转载请保留说明!

上一篇:javascript生成随机数方法汇总(javascript生成随机整数)

下一篇:每天一篇javascript学习小结(Date对象)(每天一篇日记100字)

  • 增值税税负税率
  • 运费增值税税率有5吗
  • 一般风险准备是留存收益吗
  • 财政应返还额度与财政拨款收入的关系
  • 公司向个人借款不还如何处理
  • 已收保证金
  • 一般纳税人有免征增值税吗
  • 核定征收季度核定销售额
  • 公司与公司之间借款需要交税吗
  • 委托开发票的证明怎么写
  • 退回以前年度所得税费用会计分录
  • 个人劳务报酬所得税率表
  • 职工食堂支出明细表
  • 设备税款
  • 货代行业红字冲正发票怎么做凭证
  • 一月份的工作日是多少天
  • 转让无形资产所有权计入什么科目
  • 以旧换新的销售方式怎样确定销售额
  • 收益性支出包括哪些科目
  • 房地产项目代建模式
  • 王者荣耀进不去怎么回事最新
  • 单位垫付资金如何做账
  • 以非现金资产清偿全部债务
  • 员工的餐费补助怎么算
  • 怎么修改wifi密码视频教程
  • 建筑企业被靠挂靠怎么办
  • 赔偿款支出会计处理
  • 工程竣工结算和决算的区别
  • ekb install
  • 高新技术企业加计5%
  • windows默认网关应该设置为的地址
  • php中md5函数
  • 餐费发票怎么做账务处理
  • 流转税的计税依据有哪些
  • laravel auth:api
  • 递归 php
  • 什么是市盈率和市净率,谈谈你对两个指标的理解
  • openai发布时间
  • 异步函数执行顺序
  • 房地产企业预缴增值税会计处理
  • 金税盘怎么取消
  • 销售鸡蛋免增值税吗
  • 发票备注栏填写样板
  • 没有取得发票就把车卖了违法吗
  • 退税没有退的部分怎么做账
  • 普通发票上的银行账户有规定吗
  • 文化事业建设费2023年是否减免了
  • 劳务外经证预缴税款
  • 核销应收帐款分录
  • 诉讼财产保全保函
  • 二手市场机械设备
  • 结转周转材料成本
  • 财政授权支付的概念
  • 企业代理社保
  • 收到伙食费的会计处理
  • win mysql
  • Win9传闻汇总:通知中心+免费下载+手势功能等
  • 在windows操作中
  • 如何设置微信语音来电铃声
  • 手动防止Ping攻击方法(无需防火墙)
  • ibm文件是什么意思
  • win7不能运行应用程序的方法
  • window10 系统安装
  • windows 8 build
  • windows8.1升级到windows10
  • win8的桌面文件在哪里
  • w7系统序列号怎么查
  • 欢迎使用本公司智能语音电动车
  • fragment的replace方法
  • 深入理解计算机系统
  • jquery控制css样式
  • python怎么用数组
  • javascript生成随机整数
  • opencvandroid开发实战
  • 柜台申报税务流程图
  • 扣税1.5万
  • 国家税务总局发票查验平台网络异常
  • 经营租赁属于什么费用
  • 教育培训行业的发展
  • 不想订亲
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设