前言

本文的文字及图片来源于网络,仅供学习、交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理。

作者: Python应用宝典

PS:如有需要Python学习资料的小伙伴可以加点击下方链接自行获取

http://note.youdao.com/noteshare?id=3054cce4add8a909e784ad934f956cef

原理简介

在我们今天的教程中,将用到即时通讯的概念,即时通讯允许两人或多人同时使用网络传递文字信息、文字、语音等。即时通讯一般都基于socket连接,socket连接可用于发送或接受数据,一般的组合形式是IP+端口号。

也就是说,在我们的例子中,聊天的双方,由一方要承担“服务器 ” 的责任,维持一个socket服务器,等待连接进入;另一方则是“客户端”,在服务器端维持等待状态时即可发送请求,建立连接。

当你和ta想进入“小黑屋 ” 里聊天的时候,只有有一方充当服务器,另一方充当客户端即可,作为“服务器端 ” 的那个人,在微信中将IP和端口号告诉对方,即可构建连接,在小黑屋里聊天,这个小黑屋里的数据不会被任何数据库保留(除非你自己做了一个保存的数据库)。

服务器端

聊天的时候,我们有时候会遇到双方同时发消息的情况。这种聊天方式就叫全双工聊天方式:“服务器”可向“客户端”发送消息,“客户端”也可向“服务端”发送消息,而且允许同时发送消息。

服务器端怎么实现全双工的聊天方式呢?其实很简单,只要用多线程就行了,主线程用于接收客户端的连接,连接成功后新建两个线程:一个用于发送消息,一个用于接收消息:

首先,建立socket服务器:

 import socket
import traceback
# 设定ip和端口号
host = ''
port = 51423
# 建立socket服务器
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s.bind((host,port))
s.listen()
while True:
# 等待连接
try:
clientsock, clientaddr = s.accept()
except KeyboardInterrupt:
raise
except:
traceback.print_exc()
continue

其中,AF_INET指的是用IPv4进行通信,而SOCK_STREAM指的是TCP协议。端口号你可以随意设定,服务器端的IP地址默认为空即可。

在while循环中不断等待用户的连接。如果有用户连接成功了,我们将进入下一步,分别建立发送和接受线程:

 # 建立接收线程
t = _thread.start_new_thread(processRecv, (clientsock,)) # 建立发送线程
r = _thread.start_new_thread(processSend, (clientsock,))

clientsock就是我们得到的socket连接,processRecv和processSend分别用于处理接受信息和处理发送信息:

 import _thread
def processRecv(clientsock):
"""
接受消息
:param clientsock: 客户端的socket连接
"""
while True:
data = clientsock.recv(4096)
if not len(data):
break
print (data.decode('utf-8'))
clientsock.close() def processSend(clientsock):
"""
发送消息
:param clientsock: 客户端的socket连接
"""
while True:
data = input("> ")
data = data
clientsock.sendall(data.encode('utf-8'))
clientsock.close()

有个小细节要注意,socket连接的sendall函数只支持bytes类型的数据,所以我们要encode('utf-8')。

服务端的所有代码就这样,没错,就是这么简单。

客户端

客户端则更简单,主线程本身设定为接受消息,那么我们只需要多一个线程用于发送消息即可。客户端的全部代码如下:

 import _thread
import sys
from socket import * def send_message(tcpCliSock):
"""
发送信息
:param tcpCliSock: 与服务端的socket连接
"""
while True:
message = input('> ')
if not message:
break
tcpCliSock.send(message.encode('utf-8')) tcpCliSock.close() if(len(sys.argv) < 3):
HOST = 'localhost'
PORT = 51423
else:
HOST = sys.argv[1]
PORT = int(sys.argv[2]) BUFSIZ = 1024
ADDR = (HOST,PORT) tcpCliSock = socket(AF_INET,SOCK_STREAM)
tcpCliSock.connect(ADDR) # 建立发送消息的线程
s = _thread.start_new_thread(send_message, (tcpCliSock,)) while True:
rdata = tcpCliSock.recv(BUFSIZ)
if not rdata:
break
print (rdata.decode('utf-8')) tcpCliSock.close()

其中,HOST部分填写对方的IP,PORT部分填写端口号。sys.argv用于通过参数输入这两个值,比如我们将客户端文件命名为:client.py, 在cmd中输入:

python client.py 127.0.0.1 51423

能直接传入参数执行脚本,除此之外,其他部分和服务端其实差不多。注意把接受到的数据decode一下(因为我们发的时候encode了)。

最新文章

  1. Angular企业级开发(5)-项目框架搭建
  2. core Bluetooth(蓝牙4.0)
  3. c#后台进行窗体切换的方法
  4. 《Google想出了一个决定人员晋升的算法,然后就没有然后了......》有感
  5. spoj687 后缀数组重复次数最多的连续重复子串
  6. H5版俄罗斯方块(3)---游戏的AI算法
  7. Codeforces Round #276 (Div. 1) B. Maximum Value 筛倍数
  8. MyBatis之八:需要说明的几个java api的生命周期以及封装
  9. h2database源码浅析:集群
  10. js简单实现删除记录时的提示效果
  11. Android 自定义dialog(AlertDialog的修改样式)
  12. hdu 1011 Starship Troopers_树状dp
  13. React配合Webpack实现代码分割与异步加载
  14. tcp/ip详解 卷1 -- 链路层
  15. SQL Server数据库————增删改查
  16. linux操作命令之帮助命令
  17. High Availability手册(1): 环境
  18. 向量空间模型(Vector Space Model)的理解
  19. 阿里云ECS centos7配置tomcat
  20. element-ui中用el-dialog+el-table+el-pagination实现文件默认选中且在分页的条件下有记忆功能

热门文章

  1. swoole怎么保持不掉线
  2. go语言的json
  3. SAP 表汇总
  4. C#之Form表单认证
  5. Slickflow.NET 开源工作流引擎快速入门之二: 简单并行分支流程代码编写示例
  6. centos7.2安装及管理docker
  7. vue-svgicon基本使用
  8. ant 打包脚本
  9. arcgis api 4.x for js 地图加载多个气泡窗口展示(附源码下载)
  10. cell右侧的状态(accessoryType)