第一个tornado程序,经典的hello.py

代码

import tornado.httpserver
import tornado.ioloop
import tornado.options
import tornado.web

from tornado.options import define, options
define("port", default=8000, help="run on the given port", type=int)

class IndexHandler(tornado.web.RequestHandler):
    def get(self):
        greeting = self.get_argument('greeting', 'Hello')
        self.write(greeting + ', friendly user!')

if __name__ == "__main__":
    tornado.options.parse_command_line()
    app = tornado.web.Application(handlers=[(r"/", IndexHandler)])
    http_server = tornado.httpserver.HTTPServer(app)
    http_server.listen(options.port)
    tornado.ioloop.IOLoop.instance().start()

测试

(1)在命令行执行:python hello.py
(2.1)在浏览器打开:http://ip:8000
得到:Hello, friendly user!
(2.2)在浏览器打开:http://ip:8000?greeting=Welcome
得到:Welcome, friendly user!

代码解析

from tornado.options import define, options
define("port", default=8000, help="run on the given port", type=int)

Tornado包括了一个有用的模块(tornado.options)来从命令行中读取设置。我们在这里使用这个模块指定我们的应用监听HTTP请求的端口。它的工作流程如下:如果一个与define语句中同名的设置在命令行中被给出,那么它将成为全局options的一个属性。如果用户运行程序时使用了–help选项,程序将打印出所有你定义的选项以及你在define函数的help参数中指定的文本。如果用户没有为这个选项指定值,则使用default的值进行代替。Tornado使用type参数进行基本的参数类型验证,当不合适的类型被给出时抛出一个异常。因此,我们允许一个整数的port参数作为options.port来访问程序。如果用户没有指定值,则默认为8000。

class IndexHandler(tornado.web.RequestHandler):
    def get(self):
        greeting = self.get_argument('greeting', 'Hello')
        self.write(greeting + ', friendly user!')

这是Tornado的请求处理函数类。当处理一个请求时,Tornado将这个类实例化,并调用与HTTP请求方法所对应的方法。在这个例子中,我们只定义了一个get方法,也就是说这个处理函数将对HTTP的GET请求作出响应。我们稍后将看到实现不止一个HTTP方法的处理函数。

greeting = self.get_argument('greeting', 'Hello')

Tornado的RequestHandler类有一系列有用的内建方法,包括get_argument,我们在这里从一个查询字符串中取得参数greeting的值。(如果这个参数没有出现在查询字符串中,Tornado将使用get_argument的第二个参数作为默认值。)

self.write(greeting + ', friendly user!')

RequestHandler的另一个有用的方法是write,它以一个字符串作为函数的参数,并将其写入到HTTP响应中。在这里,我们使用请求中greeting参数提供的值插入到greeting中,并写回到响应中。

if __name__ == "__main__":
    tornado.options.parse_command_line()
    app = tornado.web.Application(handlers=[(r"/", IndexHandler)])

这是真正使得Tornado运转起来的语句。首先,我们使用Tornado的options模块来解析命令行。然后我们创建了一个Tornado的Application类的实例。传递给Application类init方法的最重要的参数是handlers。它告诉Tornado应该用哪个类来响应请求。马上我们讲解更多相关知识。

http_server = tornado.httpserver.HTTPServer(app)
http_server.listen(options.port)
tornado.ioloop.IOLoop.instance().start()

从这里开始的代码将会被反复使用:一旦Application对象被创建,我们可以将其传递给Tornado的HTTPServer对象,然后使用我们在命令行指定的端口进行监听(通过options对象取出。)最后,在程序准备好接收HTTP请求后,我们创建一个Tornado的IOLoop的实例。

参数handlers

app = tornado.web.Application(handlers=[(r"/", IndexHandler)])

这里的参数handlers非常重要,值得我们更加深入的研究。它应该是一个元组组成的列表,其中每个元组的第一个元素是一个用于匹配的正则表达式,第二个元素是一个RequestHanlder类。在hello.py中,我们只指定了一个正则表达式-RequestHanlder对,但你可以按你的需要指定任意多个。

`tornado.tcpserver.TCPServer` 是 Tornado 框架中的一个类,用于实现 TCP 服务器的功能。它继承自 `tornado.netutil.TCPServer` 类,提供了对 TCP 连接的监听、接收和处理等功能。 `TCPServer` 的初始化方法如下: ```python class tornado.tcpserver.TCPServer(io_loop=None, ssl_options=None, max_buffer_size=None, read_chunk_size=None, max_idle_time=None, connect_timeout=None, max_body_size=None) ``` 其中,参数含义如下: - `io_loop`: 事件循环对象,用于处理 TCP 连接的事件和回调。如果不指定,将使用默认的事件循环对象。 - `ssl_options`: SSL/TLS 配置选项,用于开启加密传输。默认为 `None`,表示不开启 SSL/TLS 加密传输。 - `max_buffer_size`: 读取数据时的最大缓冲区大小,默认为 `None`,表示不限制缓冲区大小。 - `read_chunk_size`: 读取数据时的缓冲区大小,默认为 64KB。 - `max_idle_time`: 连接的最大空闲时间,超过该时间未收到数据,则关闭连接,默认为 `None`,表示不限制空闲时间。 - `connect_timeout`: 连接的最大超时时间,默认为 `None`,表示不限制超时时间。 - `max_body_size`: 读取消息体的最大大小,默认为 `None`,表示不限制消息体大小。 `TCPServer` 类还提供了以下常用方法: - `listen`: 监听指定的端口,并启动服务器。 - `stop`: 停止服务器,关闭所有连接。 - `handle_stream`: 处理新的 TCP 连接。 例如,下面的示例代码实现了一个简单的 TCP 服务器: ```python import tornado.ioloop import tornado.tcpserver import tornado.gen class MyTCPServer(tornado.tcpserver.TCPServer): async def handle_stream(self, stream, address): print(f"New connection from {address}") while True: try: data = await stream.read_until(b"\n") print(f"Received {data.decode().strip()} from {address}") await stream.write(data.upper()) except Exception as e: print(f"Error: {e}") break if __name__ == "__main__": server = MyTCPServer() server.listen(8888) tornado.ioloop.IOLoop.current().start() ``` 在上面的代码中,我们定义了一个 `MyTCPServer` 类,继承自 `tornado.tcpserver.TCPServer`。在 `handle_stream` 方法中,我们处理了新的 TCP 连接,并循环读取客户端发送的数据,并将其转换为大写后发送回客户端。最后,我们通过 `server.listen(8888)` 启动了 TCP 服务器,监听 8888 端口,并通过 `tornado.ioloop.IOLoop.current().start()` 运行了事件循环。 需要注意的是,在 `handle_stream` 方法中,我们使用了 `await` 关键字来等待客户端发送的数据,并异步地处理数据。这样可以避免在处理 TCP 连接时阻塞事件循环,提高服务器的并发性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值