Python Socket编程快速入门
2019-02-22T10:02:10
1761
0
0
#### 本文中,我们将以Python为例介绍Socket网络编程。
## Socket网络编程介绍
socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求。
socket起源于Unix,而Unix/Linux基本哲学之一就是“一切皆文件”,对于文件用打开、读写、关闭模式来操作。socket就是该模式的一个实现,socket即是一种特殊的文件,一些socket函数就是对其进行的操作(读/写IO、打开、关闭)。
socket和file的区别:
file模块是针对某个指定文件进行打开、读写、关闭。
socket模块是针对 服务器端 和 客户端Socket 进行打开、读写、关闭。
Socket的英文原义是“孔”或“插座”。Socket正如其英文原意那样,像一个多孔插座。一台主机犹如布满各种插座的房间,每个插座有一个编号,有的插座提供220伏交流电, 有的提供110伏交流电,有的则提供有线电视节目。 客户软件将插头插到不同编号的插座,就可以得到不同的服务。
简单来说:**两个程序通过“网络”交互数据就使用socket,它只负责两件事:建立连接,传递数据。**
一个完整的socket通信流程大致如下图所示:
![title](/static/files/591/5989cee6e519f50ef7000031/0/images/4594731667f656c31baaa95fe4b33134.png)
## 服务端实现
```python
#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author: Missshi
import socket
import subprocess #导入执行命令模块
ip_port=('127.0.0.1',9999) #定义元祖
#买手机
s=socket.socket() #绑定协议,生成套接字
s.bind(ip_port) #绑定ip+协议+端口:用来唯一标识一个进程,ip_port必须是元组格式
s.listen(5) #定义最大可以挂起链接数
#等待电话
while True: #用来重复接收新的链接
conn,addr=s.accept() #接收客户端胡的接请求,返回conn(相当于一个特定胡链接),addr是客户端ip+port
conn.sendall(bytes('Hello everyone.'))
#收消息
while True: #用来基于一个链接重复收发消息
try: #捕捉客户端异常关闭(ctrl+c)
recv_data=conn.recv(1024) #收消息,阻塞
if len(recv_data) == 0:break #客户端如果退出,服务端将收到空消息,退出
#发消息
p=subprocess.Popen(str(recv_data),shell=True,stdout=subprocess.PIPE) #执行系统命令
res=p.stdout.read() #获取标准输出
if len(res) == 0: #执行错误命令,标准输出为空,
send_data='cmd err'
print(str)
send_data=bytes(str(res))
#解决粘包问题
ready_tag='Ready|%s' %len(send_data)
conn.send(bytes(ready_tag)) #发送数据长度
feedback=conn.recv(1024) #接收确认信息
feedback=str(feedback)
if feedback.startswith('Start'):
conn.send(send_data) #发送命令的执行结果
except Exception:
break
#挂电话
conn.close()
```
其中,Server端执行的操作是接收Client端传入的数据作为操作命令。并且在本机上执行该操作命令。同时,将操作命令得到的标准输出返回给客户端。
## 客户端实现
客户端实现如下:
```python
#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author: Jason Wang
import socket
ip_port=('127.0.0.1',9999)
#买手机
s=socket.socket()
#拨号
s.connect(ip_port) #链接服务端,如果服务已经存在一个好的连接,那么挂起
welcom_msg = s.recv(200).decode() #获取服务端欢迎消息
print(welcom_msg)
while True: #基于connect建立的连接来循环发送消息
send_data=input(">>: ").strip() # 接收用户输入命令
if send_data == 'exit':break
if len(send_data) == 0:continue
s.send(bytes(send_data))
#解决粘包问题
ready_tag=s.recv(1024) #收取带数据长度的字节:Ready|9998
ready_tag=str(ready_tag)
if ready_tag.startswith('Ready'):#Ready|9998
msg_size=int(ready_tag.split('|')[-1]) #获取待接收数据长度
start_tag='Start'
s.send(bytes(start_tag)) #发送确认信息
#基于已经收到的待接收数据长度,循环接收数据
recv_size=0
recv_msg=b''
while recv_size < msg_size:
recv_data=s.recv(1024)
recv_msg+=recv_data
recv_size+=len(recv_data)
print('MSG SIZE %s RECE SIZE %s' %(msg_size,recv_size))
print(str(recv_msg))
#挂电话
s.close()
```
## 实战验证
首先,启动Server端服务:
```
python server.py
```
接下来,启动客户端服务:
```
python client.py
```
此时,我们可以输入相关命令来执行:
![title](/static/files/591/5989cee6e519f50ef7000031/81/images/7cceafe010ed2a3f53c965c9ecbb78a6.png)
0条评论
评论前请先登录
登录
注册
发表回复