010-Python-socket编程

时间:2022-01-04 23:12:33

客户端/服务器的架构

物理层:网卡,光缆,双绞线

数据链路层:包含源mac地址和目标的mac地址,通过广播通讯

网络层:跑的IP协议,IP地址可以定义到一个子网;通过ARP协议可以解析为mac地址;

传输层:提供端口 0-65535 TCP/UDP基于端口传输的

====》socket抽象层{抽象一堆接口,只要遵循socket标准,就支持TCP/UDP基于端口传输}

应用层:软件

客户端与服务端的连接:

1.服务端连接关键字注释:

import socket

server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  # 基于网络通信,是TCP协议的通讯
server.bind(("127.0.0.1", 8080)) # 开启 绑定IP以及端口
server.listen(5) # 监听;进程池的大小5 # conn 三次握手建立的连接, addr 客户端地址 端口
conn, addr = server.accept()
print("TCP 的链接", conn)
print("客户端的地址和端口", addr) # conn.recv 接收消息;1024 代表接收到 1024个字节 (二进制格式) secv
data = conn.recv(1024)
print("来自于客户端的消息: %s" % data) # 打印输出内容 # 发消息 给 客户端 send (将接收的消息转换为大写返回给客户端 二进制)
conn.send(data.upper()) conn.close() # 关闭连接
server.close() # 结束套接字

2.客户端连接关键字注释:

import socket

client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  # 基于网络通信,是TCP协议的通讯
client.connect(("127.0.0.1", 8080)) # 与服务端进行连接;
client.send("Hello".encode("utf-8")) # 客户端发消息给 服务端 send 需要二进制 encode # 客户端收取 服务端发过来的消息
data = client.recv(1024)
print("来自于服务端的数据:%s" % data) client.close() # 关闭客户端

输出内容:

# 服务端内容:
TCP 的链接 <socket.socket fd=284, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 61799)>
客户端的地址和端口 ('127.0.0.1', 61799)
来自于客户端的消息: b'Hello' # 客户端内容:
来自于服务端的数据:b'HELLO'

解决粘包问题:

1.服务端解决粘包问题:

#! /usr/bin/env python
# -*- coding: utf-8 -*-
# Date: 2017/3/4 import socket
import subprocess # 远程执行命令模块
import struct server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 基于网络通信,是TCP协议的通讯
server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 重用ip和端口,在绑定IP 端口之前 bind 之前
server.bind(("127.0.0.1", 8080)) # 开启 绑定IP以及端口
server.listen(5) # 监听;进程池的大小5 while True: # 循环连接 循环
# conn 三次握手建立的连接, addr 客户端地址 端口
conn, addr = server.accept()
print("客户端地址:", addr) while True: # 消息传输 循环
# conn.recv 接收消息;1024 代表接收到 1024个字节 (二进制格式) secv try: # 捕捉异常
cmd = conn.recv(1024) # recv 会导致堵塞 if not cmd:
break # 针对Linux 客户端断开后 进入死循环现象; print("来自于客户端的消息: %s" % cmd) # 打印输出内容
# 发消息 给 客户端 send (将接收的消息转换为大写返回给客户端 二进制) res = subprocess.Popen(cmd.decode("utf-8"), # 使用 utf-8 解码
shell=True, # 使用shell 解析
stdout=subprocess.PIPE, # 标准输出定向到 管道
stderr=subprocess.PIPE) # 标准错误输出 定向到 管道 err = res.stderr.read()
if err:
back_msg = err
else:
back_msg = res.stdout.read() #####解决粘包部分#####
conn.send(struct.pack("i", len(back_msg))) # 计算出 返回值的长度
conn.sendall(back_msg) # sendall将数据循环的返回 将得到的数据 输出返回
#####↑↑↑↑↑↑↑↑↑↑↑##### except Exception: # 如果发现 异常,直接break
break conn.close() # 关闭连接
server.close() # 结束套接字

2.客户端解决粘包问题

#! /usr/bin/env python
# -*- coding: utf-8 -*-
# Date: 2017/3/4 import socket
import struct client = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 基于网络通信,是TCP协议的通讯 client.connect(("127.0.0.1", 8080)) # 与服务端进行连接; while True:
cmd = input("输入你的命令:").strip()
if not cmd:
continue client.send(cmd.encode("utf-8"))
#####解决粘包部分#####
data = client.recv(4)
data_size = struct.unpack("i", data)[0] recv_size = 0 # 收取数据的尺寸
recv_bytes = b"" # 收到的bytes 用于拼接
while recv_size < data_size:
res = client.recv(1024) recv_bytes += res
recv_size += len(res)
#####↑↑↑↑↑↑↑↑↑↑↑##### print(recv_bytes.decode("gbk"))

当返回值过大时,通过自定义报头方式解决粘包

1.自定义报头,发送方服务端配置

#! /usr/bin/env python
# -*- coding: utf-8 -*-
# Date: 2017/3/4 import socket
import subprocess # 远程执行命令模块
import struct
import json server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 基于网络通信,是TCP协议的通讯
server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 重用ip和端口,在绑定IP 端口之前 bind 之前
server.bind(("127.0.0.1", 8080)) # 开启 绑定IP以及端口
server.listen(5) # 监听;进程池的大小5 while True: # 循环连接 循环
# conn 三次握手建立的连接, addr 客户端地址 端口
conn, addr = server.accept()
print("客户端地址:", addr) while True: # 消息传输 循环
# conn.recv 接收消息;1024 代表接收到 1024个字节 (二进制格式) secv try: # 捕捉异常
cmd = conn.recv(1024) # recv 会导致堵塞 if not cmd:
break # 针对Linux 客户端断开后 进入死循环现象; print("来自于客户端的消息: %s" % cmd) # 打印输出内容
# 发消息 给 客户端 send (将接收的消息转换为大写返回给客户端 二进制) res = subprocess.Popen(cmd.decode("utf-8"), # 使用 utf-8 解码
shell=True, # 使用shell 解析
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) # 将字典依照 json 格式存储
head_bytes = head_json.encode("utf-8") # 将json 格式 解码为 二进制bytes格式 conn.send(struct.pack("i", len(head_bytes))) # 第一次发:将计算的报头head_bytes长度发送给客户端
conn.send(head_bytes) # 第二次发:发送报头的数据 conn.sendall(back_msg) # 第三次发:sendall将数据循环的返回 将得到的数据 输出返回 except Exception: # 如果发现 异常,直接break
break
# ######↑↑↑↑↑↑↑↑↑↑↑↑↑ conn.close() # 关闭连接
server.close() # 结束套接字

2.自定义报头,接收客户端配置

#! /usr/bin/env python
# -*- coding: utf-8 -*-
# Date: 2017/3/4 import socket
import struct
import json client = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 基于网络通信,是TCP协议的通讯
client.connect(("127.0.0.1", 8080)) # 与服务端进行连接; while True:
cmd = input("输入你的命令:").strip()
if not cmd:
continue client.send(cmd.encode("utf-8")) # 收取 服务端返回的数据:分三次收取
# 1.收报头的长度
head = client.recv(4) # 收取报头 二进制格式"b'\x11\x00\x00\x00'"
head_size = struct.unpack("i", head)[0] # 取出报头长度 int "17" # 2.收报头:根据报头的长度
head_bytes = client.recv(head_size) # 根据报头长度head_size 收取报头 二进制格式"b'{"data_size": 61}'"
head_json = head_bytes.decode("utf-8") # 将bytes的格式解码为 json的字符串 "'str'{"data_size": 61}" head_dic = json.loads(head_json) # 将json字符串转换为 字典 "'dict'{'data_size': 61}"
data_size = head_dic["data_size"] # 取出真实数据的长度 "61" # 3.根据长度-收取真实数据
recv_size = 0 # 收取数据的尺寸
recv_bytes = b"" # 收到的bytes 用于拼接
while recv_size < data_size:
res = client.recv(1024) recv_bytes += res
recv_size += len(res)
print(recv_bytes.decode("gbk"))

010-Python-socket编程的更多相关文章

  1. Python Socket 编程——聊天室示例程序

    上一篇 我们学习了简单的 Python TCP Socket 编程,通过分别写服务端和客户端的代码了解基本的 Python Socket 编程模型.本文再通过一个例子来加强一下对 Socket 编程的 ...

  2. python&sol;socket编程之粘包

    python/socket编程之粘包 粘包 只有TCP有粘包现象,UDP永远不会粘包. 首先需要掌握一个socket收发消息的原理 发送端可以是1k,1k的发送数据而接受端的应用程序可以2k,2k的提 ...

  3. PYTHON SOCKET编程简介

    原文地址: PYTHON SOCKET编程详细介绍   Python 提供了两个基本的 socket 模块. 第一个是 Socket,它提供了标准的 BSD Sockets API. 第二个是 Soc ...

  4. python socket编程笔记

    用python实现一个简单的socket网络聊天通讯 (Linux --py2.7平台与windows--py3.6平台) 人生苦短之我用Python篇(socket编程) python之路 sock ...

  5. &lbrack;Python&lowbar;7&rsqb; Python Socket 编程

    0. 说明 Python Socket 编程 1. TCP 协议 [TCP Server] 通过 netstat -ano 查看端口是否开启 # -*-coding:utf-8-*- "&q ...

  6. Python Socket 编程示例 Echo Server

    简评:我们已经从「Python Socket 编程概览」了解了 socket API 的概述以及客户端和服务器的通信方式,接下来让我们创建第一个客户端和服务器,我们将从一个简单的实现开始,服务器将简单 ...

  7. Python Socket 编程——聊天室演示样例程序

    上一篇 我们学习了简单的 Python TCP Socket 编程,通过分别写服务端和client的代码了解主要的 Python Socket 编程模型.本文再通过一个样例来加强一下对 Socket ...

  8. python socket编程入门(编写server实例&rpar;&plus;send 与sendall的区别与使用方法

    python 编写server的步骤: 1. 第一步是创建socket对象.调用socket构造函数.如: socket = socket.socket( family, type ) family参 ...

  9. 第九章:Python高级编程-Python socket编程

    第九章:Python高级编程-Python socket编程 Python3高级核心技术97讲 笔记 9.1 弄懂HTTP.Socket.TCP这几个概念 Socket为我们封装好了协议 9.2 cl ...

  10. python socket编程详细介绍

    Python 提供了两个基本的 socket 模块. 第一个是 Socket,它提供了标准的 BSD Sockets API. 第二个是 SocketServer, 它提供了服务器中心类,可以简化网络 ...

随机推荐

  1. Nginx 和 Apache 开启目录浏览功能

    1.Nginx 在相应项目的 Server 段中的 location 段中,添加 autoindex on.例如: server { listen ; server_name www.dee.prac ...

  2. python python 入门学习之网页数据爬虫cnbeta文章保存

    需求驱动学习的动力. 因为我们单位上不了外网所以读新闻是那么的痛苦,试着自己抓取网页保存下来,然后离线阅读.今天抓取的是cnbeta科技新闻,抓取地址是http://m.cnbeta.com/wap/ ...

  3. &lbrack;Java面试二&rsqb;Java基础知识精华部分&period;

    一:java概述(快速浏览): 1991 年Sun公司的James Gosling等人开始开发名称为 Oak 的语言,希望用于控制嵌入在有线电视交换盒.PDA等的微处理器: 1994年将Oak语言更名 ...

  4. JSP的那些事儿(2)---- DWR2&period;0 的配置和使用

    JSP的那些事儿(2)----DWR2.0 的配置和使用 分类: Web开发 JAVA 2009-04-23 15:43 999人阅读 评论(0) 收藏 举报 jspdwrjavascriptserv ...

  5. FZU 2092 收集水晶 bfs&plus;记忆化搜索 or 暴力

    题目链接:收集水晶 一眼看过去,觉得是普通的bfs,初始位置有两个.仔细想了想...好像如果这样的话..........[不知道怎么说...T_T] dp[12][12][12][12][210] 中 ...

  6. python3&period;5读取网页代码,并保存

    在旧版的python中有个urllib模块,内有一个urlopen方法可打开网页,但新版python中没有了,新版的urllib模块里面只有4个子模块(error,request,response,p ...

  7. python安装json的方法;以及三种json库的区别

    python中的json解释库有好几个,不同版本使用方法不同. 常用有 json-py 与smiplejson 两个包 其中,json-py 包含json.py外,还有一个minjson,两者用法上有 ...

  8. &lbrack;七&rsqb;JavaIO之 PipedInputStream 和 PipedInputStream

    管道简介

  9. nil&sol;Nil&sol;NULL&sol;NSNull

    nil/Nil/NULL/NSNull的区别 一个简单的小例子: NSObject *obj = nil; NSLog(@"%@",obj); =>null NSObject ...

  10. Bootstrap洼地

    前面的话 这是一个轻量.灵活的组件,它能延伸至整个浏览器视口来展示网站上的关键内容.本文将详细介绍Bootstrap洼地 概述 洼地(Well)样式的效果和巨幕jumbotron样式类似,不同点是we ...