Python之旅Day8 socket网络编程
socket网络编程
Socket是网络编程的一个抽象概念。通常我们用一个Socket表示“打开了一个网络链接”,而打开一个Socket需要知道目标计算机的IP地址和端口号,再指定协议类型即可。socket服务一般分为服务端和客户端;与此同时,socket服务有基本tcp和udp的两种版本类型
TCP类型
基础版socket服务
####服务端####
import socket
phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)#买手机
'''AF_INET:地址家族; SOCK_STREAM:TCP协议''' phone.bind(('127.0.0.1',8080))#插卡 phone.listen(5) #开机(监听)
'''listen(5):等待客户端连接的最大连接数(可自定义数值)''' conn,addr=phone.accept()#接电话(3次握手建立连接)
print('tcp的连接',conn)
print('客户端的地址',addr) data = conn.recv(1024) #说话(收消息)
print('from client msg: %s' %data) conn.send(data.upper()) #发消息 conn.close()#挂电话
phone.close()#关手机 ####客户端####
import socket
client = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
'''AF_INET:地址家族; SOCK_STREAM:TCP协议''' client.connect(('127.0.0.1',8080)) #拨通电话 client.send('hello'.encode('utf-8')) #客户端发消息 data = client.recv(1024) #客户端收消息
print(data)
client.close() #关闭
循环版socket服务
####服务端####
import socket
phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)#买手机
'''AF_INET:地址家族; SOCK_STREAM:TCP协议''' phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
'''用来解决报错:ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接。''' phone.bind(('127.0.0.1',8080))#插卡 phone.listen(5) #开机(监听)
'''listen(5):等待客户端连接的最大连接数(可自定义数值)''' while True: #链接循环
conn,addr=phone.accept()#接电话(3次握手建立连接)
print('client: ',addr) while True: #通讯循环
try:
data = conn.recv(1024) #说话(收消息)
if not data:break #针对Linux,客户端断开链接的异常处理
print('from client msg: %s' %data) conn.send(data.upper()) #发消息
except Exception: #异常捕捉
break conn.close()#挂电话 phone.close()#关手机 ####客户端####
import socket
client = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
'''AF_INET:地址家族; SOCK_STREAM:TCP协议''' client.connect(('127.0.0.1',8080)) #拨通电话 while True:
msg = input('>>>: ')
if not msg:continue
client.send(msg.encode('utf-8')) #客户端发消息 data = client.recv(1024) #客户端收消息
print(data) client.close() #关闭
socketserver服务(可并发的socket)
####服务端####
import socketserver class FtpServer(socketserver.BaseRequestHandler):
def handle(self):
print(self.request) #conn
print(self.client_address) while True:
data = self.request.recv(1024)
self.request.send(data.upper()) if __name__ == '__main__':
s = socketserver.ThreadingTCPServer(('127.0.0.1',8080),FtpServer)
s.serve_forever() #链接循环就有了 ####客户端1####
# import socket
from socket import * client = socket(AF_INET,SOCK_STREAM)
client.connect(('127.0.0.1',8080)) while True:
msg = input('>>: ')
client.send(msg.encode('utf-8')) data = client.recv(1024)
print(data) ####客户端1####
# import socket
from socket import * client = socket(AF_INET,SOCK_STREAM)
client.connect(('127.0.0.1',8080)) while True:
msg = input('>>: ')
client.send(msg.encode('utf-8')) data = client.recv(1024)
print(data) """
socketserver可以实现多并发,即一个服务端同时和多个客户端进行通信
"""
远程执行
###远程执行命令-server端###
import socket
import subprocess
import struct
import json phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)#买手机
'''AF_INET:地址家族; SOCK_STREAM:TCP协议''' phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
'''用来解决报错:ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接。''' phone.bind(('127.0.0.1',8080))#插卡 phone.listen(5) #开机(监听)
'''listen(5):等待客户端连接的最大连接数(可自定义数值)''' while True: #链接循环
conn,addr=phone.accept()#接电话(3次握手建立连接)
print('client: ',addr) while True: #通讯循环
try:
cmd = conn.recv(1024) #说话(收消息)
if not cmd:break #针对Linux,客户端断开链接的异常处理
print('from client msg: %s' %cmd) res = subprocess.Popen(cmd.decode('utf-8'),
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE
)
err = res.stderr.read()
if err:
back_msg = err
else:
back_msg = res.stdout.read() #解决长度局限
#第一阶段:制作报头
head_dic = {
'data_size':len(back_msg)
}
head_json = json.dumps(head_dic)
head_bytes = head_json.encode('utf-8') '''添加解决粘包代码'''
# conn.send(struct.pack('i',len(back_msg))) #传数据的长度(有局限) #第二阶段:发送报头的长度
conn.send(struct.pack('i',len(head_bytes))) # conn.send(back_msg)
# 第三阶段:发报头
conn.send(head_bytes) #第四阶段:发真实数据
conn.sendall(back_msg) except Exception: #异常捕捉
break conn.close()#挂电话 phone.close()#关手机 ###远程执行命令-client端###
import socket
import struct
import json client = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
client.connect(('127.0.0.1',8080)) while True:
cmd= input('>>>: ').strip()
if not cmd:continue client.send(cmd.encode('utf-8'))
#解决粘包注释的改动部分(注释掉)
# res=client.recv(1024)
# print(res.decode('gbk')) '''通过定义传输数据文件大小来解决粘包'''
# data = client.recv(4)
# data_size=struct.unpack('i',data)[0] ###优化:接收头的长度
head = client.recv(4)
head_size = struct.unpack('i',head)[0] ###收报头(根据报头长度)
head_bytes = client.recv(head_size)
head_json = head_bytes.decode('utf-8')
head_dic= json.loads(head_json)
data_size = head_dic['data_size'] #取真实数据 """大数据量传输(收真实数据)"""
recv_size = 0
recv_bytes = b''
while recv_size < data_size:
res=client.recv(1024)
recv_bytes+=res
recv_size+=len(res)
print(recv_bytes.decode('gbk')) # res=client.recv(data_size)
# print(res.decode('gbk'))
Python之旅Day8 socket网络编程的更多相关文章
- Python全栈【Socket网络编程】
Python全栈[socket网络编程] 本章内容: Socket 基于TCP的套接字 基于UDP的套接字 TCP粘包 SocketServer 模块(ThreadingTCPServer源码剖析) ...
- Python面向对象进阶和socket网络编程-day08
写在前面 上课第八天,打卡: 为什么坚持?想一想当初: 一.面向对象进阶 - 1.反射补充 - 通过字符串去操作一个对象的属性,称之为反射: - 示例1: class Chinese: def __i ...
- Python面向对象进阶和socket网络编程
写在前面 为什么坚持?想一想当初: 一.面向对象进阶 - 1.反射补充 - 通过字符串去操作一个对象的属性,称之为反射: - 示例1: class Chinese: def __init__(self ...
- NO.8:自学python之路------并行socket网络编程
摘要 一到放假就杂事很多,这次的作业比较复杂,做了一个周,进度又拖了.不过结果还不错. 正文 粘包 在上一节中,如果连续发送过多数据,就可能发生粘包.粘包就是两次发送的数据粘在一起被接收,损坏了数据的 ...
- Python之路,Day8 - Socket编程进阶
Python之路,Day8 - Socket编程进阶 本节内容: Socket语法及相关 SocketServer实现多并发 Socket语法及相关 socket概念 socket本质上就是在2台 ...
- Python Socket 网络编程
Socket 是进程间通信的一种方式,它与其他进程间通信的一个主要不同是:它能实现不同主机间的进程间通信,我们网络上各种各样的服务大多都是基于 Socket 来完成通信的,例如我们每天浏览网页.QQ ...
- python之Socket网络编程
什么是网络? 网络是由节点和连线构成,表示诸多对象及其相互联系.在数学上,网络是一种图,一般认为专指加权图.网络除了数学定义外,还有具体的物理含义,即网络是从某种相同类型的实际问题中抽象出来的模型.在 ...
- Python之路【第七篇】python基础 之socket网络编程
本篇文章大部分借鉴 http://www.cnblogs.com/nulige/p/6235531.html python socket 网络编程 一.服务端和客户端 BS架构 (腾讯通软件:ser ...
- 从零开始学Python第八周:网络编程基础(socket)
Socket网络编程 一,Socket编程 (1)Socket方法介绍 Socket是网络编程的一个抽象概念.通常我们用一个Socket表示"打开了一个网络链接",而打开一个Soc ...
随机推荐
- mysql学习笔记--数据库内置函数
一.数字类 1. 生成随机数:rand() a. 随机抽取2位 select * from stuinfo order by rand() limit 2 2. 四舍五入:round(数字) 3. 向 ...
- SpringBoot集成MyBatis的分页插件 PageHelper
首先说说MyBatis框架的PageHelper插件吧,它是一个非常好用的分页插件,通常我们的项目中如果集成了MyBatis的话,几乎都会用到它,因为分页的业务逻辑说复杂也不复杂,但是有插件我们何乐而 ...
- ASP.Net的工作线程与请求队列
当 ASP.NET 接收针对页的请求时,它从线程池中提取一个线程并将请求分配给该线程. 一个普通的(或同步的)页在该请求期间保留线程,从而防止该线程用于处理其他请求.如果一个同步请求成为 I/O bo ...
- Putty6.0 提示Access denied
1.如果putty能正常使用,解决方法很简单: 只要在Putty的configuration里面Connection->SSH->Auth->GSSAPI的配置中,去掉默认的Atte ...
- Top值
业务开发中经常会用到元素或者浏览器窗口的各种top值,最近开发组件的过程中也遇到各种问题,因此决定好好总结一下. 常见的top值 scrollTop Element.scrollTop 属性可以获取或 ...
- springCloud笔记
分布式和集群的理解:比如在一个厨房有两个厨师,一个炒菜,一个洗菜,各自做不同的事情,但是却在合作,这种叫做分布式,两个都在炒菜或者都在做菜,就叫做集群. eureka的是springCloud的注册中 ...
- 在Java的Condition接口【唤醒全部线程】
在Java的Condition接口中,存在的几个方法跟Synchronized中的wait(),waitall(),wait(time ^),这个几个方法一一对应起来,但是在Lock.newCondi ...
- DWR使用总结
这两天学了下DWR,现在总结一下. DWR是方便使用AJAX连接JS和JAVA的的一个框架,把服务器端 Java 对象的方法公开给 JavaScript 代码. 如果是用dwr2.0的jar包,还 ...
- 【转】【完全开源】百度地图Web service API C#.NET版,带地图显示控件、导航控件、POI查找控件
[转][完全开源]百度地图Web service API C#.NET版,带地图显示控件.导航控件.POI查找控件 目录 概述 功能 如何使用 参考帮助 概述 源代码主要包含三个项目,BMap.NET ...
- Windows 10 专业版 长期服务版 激活
这个用小白系统之后一段时间显示要求激活,或者更改产品秘钥.网上找了许多秘钥也是没啥用,又不想用激活工具的话,可以试试用win+R 输入cmd : 依次输入:slmgr /skms kms.digibo ...