Python 协程

手动协程操作:

# pip install gevent
from greenlet import greenlet


def test():
    print('He ')
    gr2.switch()  # 切换到test2
    print('a ')
    gr2.switch()

    
def test2():
    print('is ')
    gr1.switch()
    print('student.')

    
gr1 = greenlet(test)   # 创建一个协程
gr2 = greenlet(test2)
gr1.switch()    # 切换到gr1,也就是运行它

2. 自动协程,遇到I/O自动切换

import gevent
from gevent import monkey; monkey.patch_all()   # 导入monkey,给所有支持的模块打上补丁,变成非阻塞模块,也就是使其能够进行协程操作


def test1(n):
    print("I'm doing my homework.")
    gevent.sleep(n)          # 模拟I/O操作,gevent自动切换
    print('Keep doing my homework.')


def test2(n):
    print("I have no time to watch TV.")
    gevent.sleep(n)
    print('Stop thinking!')


def test3():
    print("Life is hard.")


# 等待 greenlets 全部结束
gevent.joinall([
    gevent.spawn(test1,3),  # 创建一个新的greeenlet对象,并规划它去执行 test1(3)
    gevent.spawn(test2,1),
    gevent.spawn(test3),
])

结果:

I'm doing my homework.
I have no time to watch TV.
Life is hard.
Stop thinking!
Keep doing my homework.

3. socket 协程实例:

服务器:

import socket
import gevent
from gevent import monkey
monkey.patch_all()    # 打补丁,让所有支持的模块变成非阻塞的方法


def server(ip,port):
    s = socket.socket()   # 生成socket对象
    s.bind((ip, port))    # 绑定ip和端口
    s.listen()            # 设置监听
    while True:
        cli, addr = s.accept()              # 等待连接
        gevent.spawn(handle_request, cli)   # 对每一个连接,使用协程的方法生成greenlet


def handle_request(conn):
    try:
        while True:
            data = conn.recv(1024)    # 接收数据
            print("recv:", data)
            new = 'Hello '+ data.decode().title()
            conn.send(new.encode('utf8'))           # 返回数据
            if not data:
                conn.shutdown(socket.SHUT_WR)
    except Exception as e:
        print(e)
    finally:
        conn.close()


if __name__ == '__main__':
    server('localhost',9999)

客户端:

import socket


client = socket.socket()
client.connect(('localhost',9999))

while True:
    con = input('>>>:').strip()
    if len(con) ==0: continue
    client.send(con.encode('utf-8'))
    data = client.recv(1024)
    print(data.decode())

client.close()
posted @ 2019-12-17 13:52  wztshine  阅读(161)  评论(0编辑  收藏  举报