文档章节

Gevent异步服务类实现多姿势WEB实时展示

乐搏学院
 乐搏学院
发布于 2017/02/24 16:12
字数 1307
阅读 5
收藏 0

内置服务:

1. gevent.server.StreamServer类, 常用于创建异步TCP服务器

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

#!/usr/bin/env python

# -*- coding: utf-8 -*-

"""

#

# Authors: limanman

# OsChina: http://xmdevops.blog.51cto.com/

# Purpose:

#

"""

# 说明: 导入公共模块

import time

import gevent

from gevent.server import StreamServer

# 说明: 导入其它模块

def tcp_handler(socket, address):

    timestamp = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime())

    print '%s client(%s) connectd to server.' % (timestamp, address)

    gevent.sleep(5)

    socket.close()

if __name__ == '__main__':

    host = ''

    port = 80

    server = StreamServer((host, port), tcp_handler)

    server.serve_forever()

2. gevent.server.DatagramServer类, 常用于创建异步UDP服务器

 

扩展服务:

1. gevent.server.pywsgi类, 可用于创建支持单向实时轮询服务器, 轮询(polling),浏览器定期发送Ajax请求,服务器收到请求后立马响应且关闭连接

优点: 后端程序编写比较容易

缺点: 请求中大半无用,浪费带宽和服务器资源

实例: 适用于小型应用

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

#!/usr/bin/env python

# -*- coding: utf-8 -*-

"""

#

# Authors: limanman

# OsChina: http://xmdevops.blog.51cto.com/

# Purpose:

#

"""

# 说明: 导入公共模块

import time

from gevent.pywsgi import WSGIServer

# 说明: 导入其它模块

def ajax_endpoint(environ, start_response):

    status = '200 OK'

    header = [

        ('Content-Type''text/html'),

        # 允许跨域

        ('Access-Control-Allow-Origin''*')

    ]

    start_response(status, header)

    return str(time.time())

if __name__ == '__main__':

    host = ''

    port = 80

    server = WSGIServer((host, port), ajax_endpoint)

    server.serve_forever()

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

<!DOCTYPE html>

<html lang="zh-en">

    <head>

        <meta charset="UTF-8">

        <title>polling</title>

        <script type="text/javascript" src="http://code.jquery.com/jquery-latest.js"></script>

    </head>

    <body>

    </body>

    <script type="text/javascript">

        setInterval(function () {

            $.ajax({

                url: 'http://127.0.0.1/',

                success:function (data) {

                    $('body').append('<p>' + data + '</p>')

                }

            })

        }, 1000)

    </script>

</html>

2. gevent.server.pywsgi类, 可用于创建支持单向实时长轮询服务器, 长轮询(long-polling),客户端向服务器发送Ajax请求,服务器收到请求后hold住连接,直到有消息才返回响应信息并关闭连接,客户端处理完响应信息后再向服务器发送新的请求

优点: 在无消息的情况下不会频繁的请求,耗费资源小

缺点: 服务器hold连接会消耗资源,返回数据顺序无保证,难以管理维护

实例: WebQQ,Hi网页版,Facebook IM等

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

#!/usr/bin/env python

# -*- coding: utf-8 -*-

"""

#

# Authors: limanman

# OsChina: http://xmdevops.blog.51cto.com/

# Purpose:

#

"""

# 说明: 导入公共模块

import time

import gevent

import itertools

from gevent.pywsgi import WSGIServer

from gevent.queue import Queue, Empty

# 说明: 导入其它模块

= Queue()

def generate_data():

    cycle = itertools.cycle(xrange(1101))

    while True:

        q.put_nowait(str(cycle.next()))

        gevent.sleep(5)

def ajax_endpoint(environ, start_response):

    status = '200 OK'

    header = [

        ('Content-Type''text/html'),

        # 允许跨域

        ('Access-Control-Allow-Origin''*')

    ]

    start_response(status, header)

    while True:

        try:

            yield q.get(timeout=1)

        except Empty, e:

            # 队列为空异常返回

            return

if __name__ == '__main__':

    host = ''

    port = 80

    = gevent.spawn(generate_data)

    g.start()

    server = WSGIServer((host, port), ajax_endpoint)

    server.serve_forever()

    g.join()

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

<!DOCTYPE html>

<html lang="zh-en">

    <head>

        <meta charset="UTF-8">

        <title>polling</title>

        <script type="text/javascript" src="http://code.jquery.com/jquery-latest.js"></script>

    </head>

    <body>

    </body>

    <script type="text/javascript">

        function longPolling() {

            $.ajax({

                url: 'http://127.0.0.1/',

                success:function (data) {

                    $('body').append('<p>' + data + '</p>')

                    if(data != '100') {

                        longPolling()

                    }

                }

            })

        }

        longPolling()

    </script>

</html>

说明: 长轮询和轮询的区别在于轮询每次由Ajax发出请求,服务端处理完毕返回响应后就结束了这条连接,而长轮询由Ajax发出请求,但是发出后会阻塞在那里等待回应,服务端只需要从数据源定期取数据就好,超时后会将之前yied的值一次性返回,本次连接断开,所以返回的数据顺序以及数量是不能保证的.

 

3. gevent.server.pywsgi+geventwebsocket(pip install gevent-websocket)类, 可用于创建支持双向实时WebSockets服务器,但并不是所有的浏览器都支持WebSockets,个人更推荐socket.io,由于它封装了WebSocket,且支持多种连接方式.

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

#!/usr/bin/env python

# -*- coding: utf-8 -*-

"""

#

# Authors: limanman

# OsChina: http://xmdevops.blog.51cto.com/

# Purpose:

#

"""

# 说明: 导入公共模块

import time

import gevent

from gevent.pywsgi import WSGIServer

from geventwebsocket import WebSocketError

from geventwebsocket.handler import WebSocketHandler

# 说明: 导入其它模块

def ws_handler(environ, start_response):

    ws = environ["wsgi.websocket"]

    while True:

        try:

            ws.send(str(time.time()))

            gevent.sleep(1)

        except WebSocketError, e:

            pass

if __name__ == '__main__':

    host = ''

    port = 5000

    server = WSGIServer((host, port), ws_handler, handler_class=WebSocketHandler)

    server.serve_forever()

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

<!DOCTYPE html>

<html lang="zh-en">

    <head>

        <meta charset="UTF-8">

        <title>polling</title>

        <script type="text/javascript" src="http://code.jquery.com/jquery-latest.js"></script>

    </head>

    <body>

        <div id="conn_status">Not Connected</div>

        <div id="placeholder" style="width:600px;height:300px;"></div>

    </body>

    <script type="text/javascript">

        $(function () {

            var ws = new WebSocket('ws://127.0.0.1:5000');

            ws.onmessage = function (event) {

                $('#placeholder ').append('<p>' + event.data + '</p>')

            }

            ws.onopen = function(event) {

                $('#conn_status').html('<b>Connected</b>');

            }

            ws.onerror = function(event) {

                $('#conn_status').html('<b>Error</b>');

            }

            ws.onclose = function(event) {

                $('#conn_status').html('<b>Closed</b>');

            }

        })

    </script>

</html>

说明: 利用gevent-websocket的WebSocketHandler处理类,在服务端调用send()/receive(),可以很方便的实现Websocket客户端与服务端的实时通信,不妨尝试来一发,运维利器网页版tail -f的Websocket实现?

 

登录乐搏学院官网http://www.learnbo.com/

或关注我们的官方微博微信,还有更多惊喜哦~

 

本文出自 “满满李 - 运维开发之路” 博客,请务必保留此出处http://xmdevops.blog.51cto.com/11144840/1862737

© 著作权归作者所有

共有 人打赏支持
乐搏学院
粉丝 9
博文 526
码字总数 707467
作品 0
丰台
程序员
私信 提问
新年新项目,聊一聊最近在做的 LightIO,IO 性能的免费午餐

LightIO 是什么 gevent 是 python 中我最喜欢库之一。 只要 import gevent, 调用下 monkey patch 便会把线程替换为绿色线程,并且把标准库 socket 打上补丁。 每次 IO 操作都会由类库自动切换...

jjym
2018/01/01
0
0
Flask+Gunicorn+Gevent+Supervisor+Nginx生产环境部署

老毛病了,在用某个新框架或新架构之前,总得花时间谷歌和自己折腾一番,才能知道这个框架和架构的优缺点,才会发现自己最喜欢、用的最顺手的的一种。近期在学习python,这里记录一下自己用的...

Jx战壕
2017/07/26
0
0
谁说gevent不能做web开发的?

fastpy是一个对gevent进行易用性封装的 高性能web框架, 其充分利用了gevent协程的特点,使得原本的mysql与http请求实现自动同步转异步。 从而达到使用简单但性能高效的特点。 源代码只有800...

feimat
2016/03/16
615
0
谁说gevent不能做web的框架的?

fastpy是一个对gevent进行易用性封装的 高性能web框架, 其充分利用了gevent协程的特点,使得原本的mysql与http请求实现自动同步转异步。 从而达到使用简单但性能高效的特点。 源代码只有800...

feimat
2016/03/16
0
0
python模块介绍-gevent介绍:基于协程的网络库

python模块介绍-gevent介绍:基于协程的网络库 介绍 gevent是基于协程的Python网络库。特点: 基于libev的快速事件循环(Linux上epoll,FreeBSD上kqueue)。 基于greenlet的轻量级执行单元。 ...

磁针石
2014/01/13
0
2

没有更多内容

加载失败,请刷新页面

加载更多

OSChina 周一乱弹 —— 白掌柜说了卖货不卖身

Osc乱弹歌单(2019)请戳(这里) 【今日歌曲】 @爱漫爱 :这是一场修行分享羽肿的单曲《Moony》 手机党少年们想听歌,请使劲儿戳(这里) @clouddyy :开不开心? 开心呀, 我又不爱睡懒觉…...

小小编辑
今天
8
0
大数据教程(11.7)hadoop2.9.1平台上仓库工具hive1.2.2搭建

上一篇文章介绍了hive2.3.4的搭建,然而这个版本已经不能稳定的支持mapreduce程序。本篇博主将分享hive1.2.2工具搭建全过程。先说明:本节就直接在上一节的hadoop环境中搭建了! 一、下载apa...

em_aaron
今天
3
0
开始看《JSP&Servlet学习笔记》

1:WEB应用简介。其中1.2.1对Web容器的工作流程写得不错 2:编写Servlet。搞清楚了Java的Web目录结构,以及Web.xml的一些配置作用。特别是讲了@WebServlet标签 3:请求与响应。更细致的讲了从...

max佩恩
今天
4
0
mysql分区功能详细介绍,以及实例

一,什么是数据库分区 前段时间写过一篇关于mysql分表的的文章,下面来说一下什么是数据库分区,以mysql为例。mysql数据库中的数据是以文件的形势存在磁盘上的,默认放在/mysql/data下面(可...

吴伟祥
今天
3
0
SQL语句查询

1.1 排序 通过order by语句,可以将查询出的结果进行排序。放置在select语句的最后。 格式: SELECT * FROM 表名 ORDER BY 排序字段ASC|DESC; ASC 升序 (默认) DESC 降序 1.查询所有商品信息,...

stars永恒
今天
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部