socket模块(套接字模块)
socket模块(套接字模块)
一、最简单版本(互传一次就结束)
# 客户端
import socket
client = socket.socket()
client.connect(('127.0.0.1', 8080)) # 8080是端口号
'''
来源百度百科
'127.0.0.1'是本机回还地址,不属于任何一个有类别地址类。它代表设备的本地虚拟接口,所以默认被看作是永远不会宕掉的接口。在Windows操作系统中也有相似的定义,所以通常在安装网卡前就可以ping通这个本地回环地址。一般都会用来检查本地网络协议、基本数据接口等是否正常的。
'''
client.send(b'hello, baby! I love you!')
msg = client.recv(1024)
print(msg.decode('utf-8'))
# 服务端
import socket
server = socket.socket()
server.bind(('127.0.0.1', 8080)) # 服务端绑定ip和端口号
server.listen(5) # 半连接池,待连接(处于一个队列中)该服务端的客户端数不能超过5个
# 有一个房间,房间内是客户端和服务端,房间外有五张凳子,最多支持5个待连接客户端,再多就会报错
conn,addr = server.accept()
msg = conn.recv(1024)
print(msg.decode('utf-8'))
conn.send(b'I love you too!')
conn.close() # 关闭连接
server.close() # 关闭服务器
二、升级版(服务端不间断服务,客户端可以重复发送命令)
# 客户端
import json
import socket
import struct
client = socket.socket()
client.connect(('127.0.0.1', 8080))
while True:
cmd = input('>>>:').encode('utf-8')
print(type(cmd)) # <class 'bytes'>
if len(cmd) == 0:
continue
client.send(cmd)
# 1.接收报头
header_length = client.recv(4)
print(type(header_length)) # <class 'bytes'>
# 2.解包得到字典的长度
d_length = struct.unpack('i', header_length)[0]
print(type(d_length)) # <class 'int'>
# 3.接收字典
d = client.recv(d_length)
print(type(d)) # <class 'bytes'>
# 4.由字典拿到数据的长度
real_d = json.loads(d)
print(type(real_d)) # <class 'dict'>
data_length = real_d['info_length']
print(data_length)
print(type(data_length)) # <class 'int'>
# 5.接收真实数据
with open('file', 'a', encoding='utf-8') as f:
real_length = 0
while real_length < data_length:
data = client.recv(1024)
real_length += len(data)
f.write(data.decode('gbk'))
# 服务端
import json
import socket
import subprocess
import struct
from socket import SOL_SOCKET, SO_REUSEADDR
server = socket.socket()
# 避免因操作系统未及时回收端口造成的端口已被占用的问题
server.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)
server.bind(('127.0.0.1', 8080))
server.listen(5)
while True:
conn, addr = server.accept()
while True:
try:
cmd = conn.recv(1024).decode('utf-8')
if len(cmd) == 0:
break
obj = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
print(type(obj))
msg = obj.stdout.read() + obj.stderr.read()
print(type(msg)) # <class 'bytes'>
# 1.将数据长度放到字典中,将字典长度打包-报头
d = {'info_length': len(msg)}
json_d = json.dumps(d)
print(type(json_d)) # <class 'str'>
header = struct.pack('i', len(json_d))
print(header) # b'\x15\x00\x00\x00'
# 2.发送报头
conn.send(header)
# 3.发送字典
conn.send(json_d.encode('utf-8'))
# 4.发送真实数据
conn.send(msg)
except ConnectionResetError: #
break
conn.close()
三、TCP传输的特点
应用程序所需要的数据,都是跟所在的那台计算机内存去要(所有数据传输都遵循这个规则)
会将数据量较小的并且时间间隔比较短的数据一次性打包发送给对方
# 客户端
import socket
client = socket.socket() # 拿电话
client.connect(('127.0.0.1', 8080)) # 拨号 写的是对方的ip和port
client.send(b'hello')
client.send(b'world')
client.send(b'baby')
client.send(b'baby')
# 服务端
import socket
server = socket.socket() # 买手机 不传参数默认用的就是TCP协议
server.bind(('127.0.0.1', 8080)) # bind((host,port)) 插电话卡 绑定ip和端口
server.listen(5) # 开机 半连接池
conn, addr = server.accept() # 接听电话 等着别人给你打电话 阻塞
data = conn.recv(1024) # 听别人说话 接收1024个字节数据 阻塞
print(data)
data = conn.recv(1024) # 听别人说话 接收1024个字节数据 阻塞
print(data)
data = conn.recv(1024) # 听别人说话 接收1024个字节数据 阻塞
print(data)
# 输出内容 理论上应该是在一行显示...
b'hello'
b'worldbabybaby'
b''
四、arp协议
- 先由IP地址找到对方,然后对方将自己的mac地址发回来
五、粘包问题
假设一种情况
小明的妈妈给小明零花钱,给了他50块钱,放在了零钱罐中,但都是硬币,小明还小,两只手一次最多只能拿十个硬币,而且规定,小明只有在妈妈给他零花钱的时候才能从零钱罐中拿钱,所以,虽然妈妈每次都给小明50块的零花钱,但是小明实际上每次只能拿到10块钱。
假设另一种情况
小明和小红在玩一个游戏,游戏规则如下:小红和小明面对面坐在桌子两侧,桌子上有三个罐子,小红小明各有一个,小红手上的罐子里有三个玻璃球,规定小红将自己罐子里的球倒入桌上的空罐子,倒三次;小明在小红将球放入罐子后,拿起罐子,将里面的球倒入自己的罐子中,也倒三次。游戏开始,小红往罐子里倒了球,小明拿起罐子,将球倒入自己的罐子中,发现一次性倒出三个球。他问小红,小红说她一次性讲球全部倒进去了
这两种情况分别是,一次拿少了,一次拿多了
六、解决粘包问题
- 解决粘包问题的关键在于,每一次拿的量是未知的
- 那么我们就要设法让服务端知道我们每一次传过去的数据的长度是多少
服务端
1.先制作一个发送给客户端的字典
2.制作字典的报头
3.发送字典的报头
4.发送字典
5.再发真实数据
客户端
1.先接受字典的报头
2.解析拿到字典的数据长度
3.接收字典
4.从字典中获取真实数据的长度
5.接收真实数据
socket模块(套接字模块)的更多相关文章
- socket套接字模块及黏包现象
一.socket套接字模块 socket概念 socket层 理解socket Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口.在设计模式中,Socket其实就是一个门面模 ...
- python 全栈开发,Day33(tcp协议和udp协议,互联网协议与osi模型,socket概念,套接字(socket)初使用)
先来回顾一下昨天的内容 网络编程开发架构 B/S C/S架构网卡 mac地址网段 ip地址 : 表示了一台电脑在网络中的位置 子网掩码 : ip和子网掩码按位与得到网段 网关ip : 内置在路由器中的 ...
- socket概念 套接字
理解socket soxket因为TCP是面向流的,你发的信息如果很多很快,TCP这样就会形成黏包 Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口.在设计模式中,Socke ...
- sanic官方文档解析之Custom Protocols(自定义协议)和Socket(网络套接字)
1,Custom Protocol:自定义协议 温馨提示:自定义协议是一个高级用法,大多数的读者不需要用到此功能 通过特殊的自定义协议,你可以改变sanic的协议,自定义协议需要继承子类asyncio ...
- iOS - Socket 网络套接字
1.Socket 网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个 Socket.Socket 又称 "套接字",应用程序通常通过 "套接字& ...
- Socket称"套接字"
Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求. 二.利用Socket建立网络连接的步骤 建立Socket连接至少需要一对 ...
- Win2 Socket(套接字)相关 API
Socket(套接字) 作者信息 肖进 单位:南京中萃食品有限公司 资讯部 邮箱:xiaoj@njb.swirebev.com 电话:025-58642091 与socket有关的一些函数介绍 1.读 ...
- Python之socket(套接字)
Socket 一.概述 socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求. ...
- 网络编程(socket,套接字)
服务端地址不变 ip + mac 标识唯一一台机器 ip +端口 标识唯一客户端应用程序 套接字: 网络编程 网络编程 一.python提供了两个级别访问的网络服务 低级别的网络服务支持基本的 S ...
随机推荐
- Codeforces Round #601 (Div. 2) C League of Leesins
把每一次输入的一组数字存下来,然后把每个数字出现的组数存下来 然后找只出现过一次的数字a,那么这个数字a不是开头就是结尾,默认为开头(是哪个都无所谓),然后去找和它出现在同一组的两个数字b和c,而b和 ...
- Codeforces Round #600 (Div. 2) A. Single Push
#include<iostream> #include<cstdio> #include<cstdlib> using namespace std; int T,n ...
- Execl导出系统
前台代码: <button class="btn btn-warning" type="button" onclick="location.hr ...
- 记录 shell学习过程(5)continue break
1.continue ;i<;i++)) do ];then continue fi echo $i done # ./continue.sh12346789 2.break ;i<;i+ ...
- python hashlib 详解
1.概述 摘要算法简介 Python的hashlib提供了常见的摘要算法,如MD5,SHA1等等. 什么是摘要算法呢?摘要算法又称哈希算法.散列算法.它通过一个函数,把任意长度的数据转换为一个长度固定 ...
- python使用临时文件
# 需求 # 某项目中,我们从传感器中采集数据,没采集1G数据后,做数据分析,最终只保存分析结果 # 这样很大的临时文件如果常驻在内存,将消耗大量地内存资源,我们可以使用临时文件储存(外部储存) # ...
- Python3标准库:textwrap文本自动换行与填充
1. textwrap文本自动换行与填充 textwrap模块提供了一些快捷函数,以及可以完成所有工作的类TextWrapper.如果你只是要对一两个文本字符串进行自动或填充,快捷函数应该就够用了:否 ...
- C++-POJ1200-Crazy Search[hash]
由于已经给出字符只有NC种,故可以把子串视为一个NC进制的数,以此构造hash函数就可以了 #include <set> #include <map> #include < ...
- nginx知识学习
设备: macbook 有用的命令行: sudo nginx -t 测试nginx是否正常 sudo nginx -s reload 平滑重启 配置目录: /usr/local/etc/nginx ...
- Java爬虫学习(2)之用对象保存文件demo(1)
package com.mieba.spider; import java.util.ArrayList; import java.util.List; import java.util.Vector ...