我想知道tornado 的内部工作流程,并且看过this article ,很好,但我就是想不通
ioloop.py里面有这样一个函数
def add_handler(self, fd, handler, events):
"""Registers the given handler to receive the given events for fd."""
self._handlers[fd] = handler
self._impl.register(fd, events | self.ERROR)
那么这是什么意思?每个请求都会触发 add_handler 还是只在初始化时触发一次?
每个socket连接都会生成一个文件描述符,还是只生成一次?
ioloop 和 iostream 有什么关系?
httpserver 如何与 ioloop 和 iostream 一起工作?
有没有流程图,我看清楚了吗?
抱歉这些问题,我只是困惑
任何链接、建议、提示都有帮助。非常感谢:)
最佳答案
我看看能不能按顺序回答你的问题:
这里的 _impl 是可用的套接字轮询机制,在 Linux 上是 epoll,在 Windows 上是 select。所以 self._impl.register(fd, events | self.ERROR) 将“等待某个事件”请求传递给底层操作系统,还特别包括错误事件。
运行时,HTTPServer 将使用 IOLoop.add_handler() 注册套接字以接受连接。当连接被接受时,它们将生成更多的通信套接字,这可能还会通过 IOStream 添加事件处理程序,该处理程序也可能调用 add_handler()。因此,新的处理程序将在开始时添加,并在收到连接时添加。
是的,每个新的套接字连接都会有一个唯一的文件描述符。 HTTPServer 正在监听的原始套接字应该保留其文件描述符。文件描述符由操作系统提供。
IOLoop 处理与套接字有关的事件,例如它们是否有数据可供读取、是否可以写入以及是否发生错误。通过使用 epoll 或 select 等操作系统服务,它可以非常高效地做到这一点。
IOStream 通过单个连接处理流数据,并使用 IOLoop 异步执行此操作。例如,IOStream 可以读取尽可能多的可用数据,然后使用 IOLoop.add_handler() 等待更多数据可用。
在 listen() 上,HTTPServer 创建一个套接字,用于使用 IOLoop 监听连接。获得连接后,它使用 socket.accept() 创建一个新套接字,然后使用新的 HTTPConnection 与客户端进行通信。
HTTPConnection 使用 IOStream 向客户端传输数据或从客户端传输数据。此 IOStream 使用 IOLoop 以异步和非阻塞方式执行此操作。许多 IOStream 和 HTTPConnection 对象可以同时处于事件状态,它们都使用相同的 IOLoop。
我希望这能回答您的一些问题。我不知道一个好的结构图,但总体思路对于其他网络服务器也应该非常相似,所以可能有一些很好的信息。你链接到的那篇深入的文章看起来确实很有用,所以如果你足够理解,我建议你再试一次:)。
关于python - Tornado ioloop 和 Tornado 的工作流程是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7072701/
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返
我花了三天的时间用头撞墙,试图弄清楚为什么简单的“rake”不能通过我的规范文件。如果您遇到这种情况:任何文件夹路径中都不要有空格!。严重地。事实上,从现在开始,您命名的任何内容都没有空格。这是我的控制台输出:(在/Users/*****/Desktop/LearningRuby/learn_ruby)$rake/Users/*******/Desktop/LearningRuby/learn_ruby/00_hello/hello_spec.rb:116:in`require':cannotloadsuchfile--hello(LoadError) 最佳
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput