用socket和多进程配合多线程实现简单的tcp的聊天服务器

python中的socket套接字用于实现网络通信
tcp套接字的使用流程如下:

  1. 创建套接字对象 --- clientSocket = socket(AF_INET,SOCK_STREAM)
  2. 连接服务器,指定服务器的ip地址和端口(参数是个元组)------------clientSocket.connect(('192.168.99.173',9999))
  3. 发送信息 -------------clientSocket.send(msg.encode('utf8'))
  4. 接收信息 ----------------msg = clientSocket.recv(1024)

我们先实现一个tcp的客户端

由于socket在接收数据时默认为堵塞状态,当接收数据是进程就刮起了,这时就无法发送信息,要实现全双工的通信,需要给接收数据和发送数据各自分配一个线程。

from socket import *
from threading import Thread



def sendMsg(clientSocket):
    while True:
        msg = input('>>')
        clientSocket.send(msg.encode('utf8'))

def recvMsg(clientSocket):
    while True:
        msg = clientSocket.recv(1024)
        print('\r>>%s'%msg.decode('utf8'))

def main():

    clientSocket = socket(AF_INET,SOCK_STREAM)

    clientSocket.connect(('192.168.99.173',9999))

    tr = Thread(target=recvMsg,args=(clientSocket,)) #将套接字作为参数传给新线程,各自的线程中分别执行收,发数据
    ts = Thread(target=sendMsg,args=(clientSocket,))

    tr.start()
    ts.start()

if __name__ == '__main__':
    main()

接下来实现服务端

服务端的套接字比较特殊,因为是为客户端服务,所以要确定下来ip地址和端口号以便客户端访问,然后服务器还要能生成新的客服套接字单独为一个客户端服务,利用多进程就可以生成多个同时工作的客服socket了。
服务器的套接字工作流程如下:

  1. 创建主套接字 --------serverSocket = socket(AF_INET,SOCK_STREAM)
  2. 绑定ip和端口 --------------------- serverSocket.bind(('',9999))
  3. 改为被动套接字 ------------------------------serverSocket.listen(5)
  4. 检测客户端连接 ,若连接则返回 新的服务套接字和客户端地址ip----- newSocket,destAdr = serverSocket.accept()
  5. 每个服务套接字进行与客户端的数据交换(在各自的进程里)
# 要求
#1. 使用tcp协议通信
#2. 使用多进程配合多线程配合多线程的方式实现多个全双工的对话。

#实现
#1. 创建tcp的套接字,绑定,监听,变为被动
#2. 每收到一次请求返回新的套接字,同时创建子进程,用来单独用新套接字对话
#3. 子进程中创建两个线程,用来收发数据
import os
from socket import *
from multiprocessing import Process
from threading import Thread,local
import time

def worker(newSocket,destAdr):
    print('创建子会话进程成功..')
    #local_school = local() #创建ThreadLocal对象,用来储存各线程的局部变量
    ts = Thread(target=sendMsg,args=(newSocket,))
    tr = Thread(target=recvMsg,args=(newSocket,))

    ts.start()
    tr.start()
    
    ts.join()
    tr.join()

def sendMsg(newSocket):
    print('发送进程准备完毕!')
    print(os.getpid())
    while True:
        msg = str( os.getpid())
        newSocket.send(msg.encode('utf8'))
        time.sleep(5)
def ps_is_end():
    print('子进程结束')

def recvMsg(newSocket):
    print('接受进程准备完毕!')
    while True:
        msg = newSocket.recv(1024)
        if msg.decode('utf8')!= '':  #在ubantu中测试当客户端连接关闭时会自动发送‘’,而在windows下运行却会报错
            print('\r>>%s'%msg.decode('utf8'))
        else:
            print('%d下限了'%os.getpid())
            newSocket.close()
            break


def main():
    serverSocket = socket(AF_INET,SOCK_STREAM)

    serverSocket.bind(('',9999))

    serverSocket.listen(5)

    while True:

        newSocket,destAdr = serverSocket.accept()
        p = Process(target=worker,args=(newSocket,destAdr,))
       
        p.start()
        newSocket.close()#拷贝到了新的进程中,这里的可以删掉了
    
    serverSocket.close()


if __name__ == '__main__':
    main()

试一下效果,先启动服务器,再启动客户端,然后客户端发条信息


捕获.PNG

捕获2.PNG
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一: 网络各个协议:TCP/IP、SOCKET、HTTP 网络七层由下往上分别为物理层、数据链路层、网络层、传输层...
    iYeso阅读 1,464评论 0 13
  • 网络由下往上分为物理层、数据链路层、网络层、传输层、会话层、表示层和应用层。通过初步的了解,我知道IP协议对应于网...
    黑马饮清泉阅读 1,515评论 0 6
  • 5 置房 解放军来了,土匪胡子被消灭了,村子恢复了往日的宁静。 三爷这些年也攒了不少积蓄,解放后,经人介绍娶了杨家...
    肃山阅读 305评论 0 2
  • 《妈妈团团zhuan》? zhuan??? 赚?转?撰? 赚的是白花花的银子和和谐的家庭关系! 转的是对生活的态度...
    指甲月阅读 138评论 0 0
  • 文案:更漏短,岁月长。你看这个市井陌上,天下之处,哪朝那代哪个人不是为自已而活着! 是夜,今晚月色并不是太好。那...
    千山4499阅读 358评论 1 0