位置: 编程技术 - 正文

Node.js中的事件驱动编程详解(nodejs中的事件循环的执行顺序)

编辑:rootadmin

推荐整理分享Node.js中的事件驱动编程详解(nodejs中的事件循环的执行顺序),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:nodejs events,node.js中的事件循环,node.js中的事件循环机制是什么,node.js中的事件循环机制是什么,node.js中的事件循环是什么意思,node.js中的事件循环是什么意思,node.js中的事件循环是什么样的,node.js中的事件循环是什么,内容如对您有帮助,希望把文章链接给更多的朋友!

在传统程编程模里,I/O操作就像一个普通的本地函数调用:在函数执行完之前程序被堵塞,无法继续运行。堵塞I/O起源于早先的时间片模型,这种模型下每个进程就像一个独立的人,目的是将每个人区分开,而且每个人在同一时刻通常只能做一件事,必须等待前面的事做完才能决定下一件事做什么。但是这种在计算机网络和Internet上被广泛使用的“一个用户,一个进程”的模型伸缩性很差。管理多个进程时,会耗费很多内存,上下文切换也会占用大量资源,这些对操作系统是个很大的负担,而且随着进程数的递增,会导致系统性能急剧衰减。

多线程是个替代方案,线程是一个轻量级的进程,它会和同一个进程内的其它线程共享内存,它更像传统模型的扩展,用来并发执行多个线程,当一个线程等待I/O操作时,其它线程可以接管CPU,当I/O操作完成,前面等待的线程会被唤醒。就是说,一个运行中的线程可以被中断,然后稍候再被恢复。此外,在一些系统下线程可以在多核CPU的不同核心下并行运行。

程序员并不知道线程会在什么具体时间运行,他们必须很小心的处理共享内存的并发访问,因此必须使用一些同步原语来同步访问某个数据结构,比如使用锁或信号量,以此来强制线程以特定的行为和计划执行。那些大量依赖线程间的共享状态的应用程序,很容易就会出现一些随机性很强,难以查找的奇怪问题。

还有一种方式是使用多线程协作,由你自己负责显式的释放CPU,并把CPU时间交给其他线程使用,因为由你亲自来控制线程的执行计划,因此减小了对同步的需求,但是也提高了程序的复杂度和出错的机会,而且并没有避免多线程的那些问题。

什么是事件驱动编程

事件驱动编程(Evnet-driven programming)是一种编程风格,由事件来决定程序的执行流程,事件由事件处理器(event handler)或事件回调(event callback)来处理,事件回调是当某个特定事件发生时被调用的函数,比如数据库返回了查询结果或者用户单击了一个按钮。

回想下,在传统的堵塞I/O编程模式里,数据库查询可能像这样:上面的query函数会让当前线程或进程一直处于等待状态,直到底层数据库完成查询操作并返回。

在事件驱动模型里,这个查询会变成这样:

首先你定义了一个叫query_finished的函数,它包含了查询完成后要做的事。然后把这个函数当做参数传递给query函数,当query执行完毕会调用query_finished,而不是仅仅返回查询结果。

当你感兴趣的事件发生时会调用你定义的函数,而不是简单的返回结果值,这种编程模型就叫事件驱动编程或异步编程。这是Node一个最明显的特性,这种编程模型意味着当前进程在执行I/O操作时不会被阻塞,因此,多个I/O操作可以并行执行,当操作完成后相应的回调函数就会被调用。

事件驱动编程底层依赖于事件循环(event loop),事件循环基本上是事件检测和事件处理器触发这两种函数不断循环调用的一个结构。在每次循环里,事件循环机制需要检测发生了哪些事件,当事件发生时,它找到对应的回调函数并调用它。

事件循环只是运行在进程内的一个线程,当事件发生时,事件处理器可以单独运行并且不会被中断,也就是说:

1.在某个特定时刻最多有一个事件回调函数运行2.任何事件处理器运行时都不会被中断

有了这个,开发人员就可以不再为线程同步和并发修改共享内存这些事头疼了。

一个众所周知的秘密:

很久以前,系统编程社区的人们就知道事件驱动编程是创建高并发服务最佳方式,因为它不用保存很多上下文,因此节省了大量内存,也没有那么多上下文切换,又节省了大量执行时间。

慢慢的,这种理念渗透到了其他的平台和社区,出现了一些有名的事件循环实现,比如Ruby的Event machine,Perl的AnyEvnet,以及Python的Twisted,除了这些还有很多其它的实现和语言。

用这些框架做开发,需要学习框架相关的特定知识以及框架特定的类库,比如,使用Event Machine时,为了享受非阻塞带来的好处,你得避免使用同步类库,只能用Event Machine的异步类库。如果你使用了任何阻塞类库(比如Ruby的大多数标准库),你的服务器就失去了最佳的伸缩性,因为事件循环依然会不断地被阻塞,时不时地阻碍了I/O事件的处理。

Node.js中的事件驱动编程详解(nodejs中的事件循环的执行顺序)

Node最初就被设计成一个非阻塞I/O服务器平台,因此一般情况下,你应该期望运行在它上面的所有代码都是非阻塞的。因为JavaScript非常小,而且它不强制使用任何I/O模型(因为它没有标准的I/O类库),因此Node建立在一个很纯净的环境里,不会有什么历史遗留问题。

Node和JavaScript如何简化了异步应用程序

Node的作者Ryan Dahl,最初使用C来开发这个项目,但是发现维护函数调用的上下文太复杂,导致代码复杂度很高。然后他转用Lua,但是Lua已经有个几个阻塞的I/O类库,阻塞和非阻塞混在一起可能会让开发人员很迷惑并因此阻碍了很多人构建可伸缩的应用,于是Lua也被Dahl抛弃了。最后他转向了JavaScript,JavaScript中的闭包及第一级对象的函数,这些特性使JavaScript非常适合用作事件驱动编程。JavaScript的魔力是让Node如此流行的一个主要原因。

什么是闭包

闭包可以理解为一个特殊的函数,但是它可以继承并访问它自身被定义的那个作用域里的变量。当你将一个回调函数作为参数传递给另外一个函数时,它稍候会被调用,神奇的是,这个回调函数被稍候调用时,它居然记住了它自身定义所在的那个上下文以及父上下文里的变量,而且还可以正常访问它们。这个强大的特性是Node成功的核心。

下面的例子将展示在Web浏览器里JavaScript闭包是如何工作的。假如,你要监听一个按钮的单机事件,你可以这样做:

使用jQuery时是这样:

JavaScript里,函数是第一类对象,就是说你可以把函数当作参数来传递给其他函数。上面的两个例子,前者把一个函数赋值给另一个函数,后者把函数作为参数传递给另一个函数,单击事件的处理函数(回调函数)可以访问函数定义所在代码块下的每个变量,在这个例子里,它可以访问在它父闭包内定义的clickCount变量。

clickCount变量处在全局作用域(JavaScript里最外层的作用域),它保存了用户点击按钮的次数,通常在全局作用域下存储变量是个坏习惯,因为那样很容易跟其他代码冲突,你应该把变量放在使用它们的本地作用域里。大多时候,只用把代码用一个函数包装起来,等于另外创建了闭包,这样就可以很容易避免污染全局环境,就像这样:

注意:上面代码的第七行,定义了一个函数后立刻调用它,这是JavaScript里一个常见的设计模式:通过创建函数来创建一个新的作用域。

闭包如何帮助异步编程

在事件驱动编程模型里,先编写事件发生后将要运行的代码,然后把这些代码放到一个函数里,最后把这个函数当作参数传递给调用者,稍后由调用者函数调用。

在JavaScript里,一个函数并不是个孤立的定义,它同时会记住自己被声明的那个作用域的上下文,这种机制让JavaScript的函数可以访问函数定义所在那个上下文及父上下文里的所有变量。

当你把一个回调函数当作参数传递给调用者后,这个函数就会在稍后的某个时刻被调用。即使定义回调函数的那个作用域已经结束,在回调函数被调用时,它依然能够访问这个已结束的作用域及其父作用域里的所有变量。像最后那个例子,回调函数在jQuery的click()内部被调用,它却依然能访问clickCount变量。

前面展现了闭包的神奇之处,把状态变量传递给一个函数就可以让你不用维护状态就能进行事件驱动编程,JavaScript的闭包机制会帮你维护它们。

小结

事件驱动编程是一种通过事件触发来决定程序执行流程的编程模型。程序员为他们感兴趣的事件注册回调函数(通常被称作事件处理器),然后系统在事件发生时调用已注册的事件处理器。这种编程模型有很多传统阻塞编程模型所不具备的优势,以前要实现类似的特性,就必须使用多进程/多线程才行。

JavaScript是种强大的语言,因为它的第一类型对象的函数和闭包特性,让它很适合事件驱动编程。

Node.js安装教程和NPM包管理器使用详解 年的JSCOnf大会上,一个叫RyanDahl的年轻程序员向人们展示了一个他正在做的项目,一个基于GoogleV8引擎的JavaScript运行平台,它提供了一套事件循环和

推荐 款优秀的高性能 Node.js 开发框架 Node.js框架可以帮助你开发出更友好的web应用。Node.js框架的许多特性和功能也能开发出大型的web应用。下面我们来介绍款最好的Node.js框架,帮助开发

使用node.js半年来总结的 条经验 先不说房价,堵车,雾霾。。。先说说我这半年使用Node.js的经验吧。。。都是工作上遇到的问题,血的教训。。1.精确版本号一定要精确到具体版本号

标签: nodejs中的事件循环的执行顺序

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

上一篇:Node.js文件操作详解(node.js document)

下一篇:Node.js安装教程和NPM包管理器使用详解(node.js安装教程详细)

  • 劳务公司一般纳税人可以抵扣进项税吗
  • 税收分类编码是唯一的吗
  • 进项税多久有效
  • 哪些福利费不税前扣除
  • 备用金可以不要发票吗
  • 生产企业发票单位
  • 政府预算年度
  • 住宿发票税率有哪4种
  • 不动产折旧年限的规定
  • 现金形式发放的福利需要扣税吗
  • 小企业周转材料报废残值回收应计入管理费用
  • 企业提供建筑服务,应向建筑服务发生地
  • 股票价值的计算公式财务管理
  • 先进制造业企业按照当期可抵扣进项税额
  • 利税总额为负数如何表述出来
  • 企业所得税退税怎么做账务处理
  • 税控盘技术服务费可以抵税吗
  • 任务栏图标调大了怎么办
  • 忘记电脑开机密码怎么打开电脑
  • 营改增后的增值税税目税率表
  • php基础入门教程
  • 最早的操作系统被称为什么操作系统
  • 买支票需要带什么章
  • 委托银行收款的会计分录怎么写
  • 跨的基本功有哪些
  • uniapp示例
  • 微信小程序解锁安全吗
  • 机器学习中的数学——距离定义(八):余弦距离(Cosine Distance)
  • python 微信红包
  • 新的企业所得税
  • 小规模纳税人企业所得税优惠政策最新2023
  • 准予扣除业务招标的情形
  • 汇算清缴的所得税怎么做账
  • 账实不符的后果和对策
  • 一般纳税人补交印花税买卖合同
  • 期间费用指哪些费用
  • 哪些发票可以抵扣?
  • 小微企业利润表数据
  • 购买股票的交易费用计入入账价值吗
  • 企业记账的相关规定
  • 发票入账的管理规定
  • 核定征税是什么意思
  • 以非现金资产抵偿债务
  • 少数股东损益会出现负数吗
  • 计提的管理费用要结转吗
  • 有限合伙企业协议
  • 不是销售性质的工作
  • 产业扶持资金管理办法
  • 付款申请单如何管理
  • 国税代开专票缴纳城建税分录?
  • 所有者权益科目有哪些
  • 国企背书是什么意思
  • mysql 5.7启动
  • win10 mobile 预览10240
  • XP系统怎么删除密码
  • 如何修改windows注册表
  • 不用光盘不用u盘怎么安装系统
  • Ubuntu 16.04 LTS正式发布,看看有哪些更新
  • centos7配置ssh免密码登录
  • centos 安装7z
  • mac图片格式怎么转换jpeg
  • win8.1怎么关闭更新
  • win8.1桌面图标消失
  • javascript学习指南
  • node.js配置
  • git 登陆用户
  • python中yield用法
  • android遇到的难题,怎么解决的
  • javascript例子
  • [置顶]公主大人接下来是拷问时间31
  • javascript异步编程
  • python文法
  • google it automation with python
  • 青岛市国家税务局官方网站
  • 深圳国家税务局赵雨婷处长
  • 安康国税局电话号码
  • 南通买房首付规定
  • 内蒙古一般纳税人查询网
  • 天津车辆购置税网上办税
  • 收购晾晒烟叶,支付价款20万元,支付价外补贴2万元
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

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

    友情链接: 武汉网站建设