注意 socket通信默认的情况下是属于阻塞形式通信,在recv与accept函数会进行阻塞

  1.客户端正常退出的情况下,server端的con的通道会正常的关闭,recv直接赋值为空

  2.在windows下,使用pycharm的强制终端客户端,会造成,con通道的破坏,造成服务器端的异常(可以通过捕获异常来避免),在linux程序端,不会出现该问题

  3.在发送数据的时候,send为空的情况下,在recv下,默认就是不处理,还处在阻塞的状态。 因此规定不能发送为空

  4.socket通信的在py3里面,只能发送bytes类型  str-->bytes: bytes(str,"utf-8")

                       bytes-->str:str(bytes"utf-8")

一.基础代码

server端:

import socket
server = socket.socket()
server.bind(("127.0.0.1",6969))
server.listen() con,addr=server.accept()
while True:
data=con.recv(1024)
data_uper=str(data,"utf-8").upper()
con.send(bytes(data_uper,"utf-8"))

client端

import socket
client = socket.socket()
client.connect(('127.0.0.1',6969))
x=input(">>:")
client.send(bytes("good 你好",encoding="utf-8"))
data=client.recv(1024)
print(str(data,"utf-8"))

二.不间断聊天代码实现:

server

import socket
server = socket.socket()
server.bind(("127.0.0.1",6969))
server.listen()
while True:
con,addr=server.accept()
print(addr)
while True:
try:
data=con.recv(1024)#在程序正常退出,也就是con的tcp连接正常中断,data直接为0执行下一步
except Exception as e:
break #防止在windows下,强制关闭,con通道错误而报错,linux下不用这样
if not data:break
print(str(data,"utf8"))
x=input(">:")
con.send(bytes(x,"utf-8"))

client

import socket
client = socket.socket()
client.connect(('127.0.0.1',6969))
while True:
x=input(">>:")
if not x:continue #解决在阻塞模型下,直接回车卡住的bug
client.send(bytes(x,encoding="utf-8"))
data=client.recv(1024)
print(str(data,"utf-8"))

三.实现类似ssh的命令。实现大数据发送的基础以及粘包的问题:

在send的时候,缓存区会不断的发送数据,但是recv接受的数据大小每次都有界限。也就是说每次发送的数据过量,一次recv是接受不完的,因为调用了解决办法:

  1.在发送端,先发送要发送的数据的大小

  2.在接受端,通过接受的数据 与 接受的数据大小 进行比较而接入不断的接受

    坑:1.int 与 bytes 不能直接转换。   2.len的大小  在str类型 与 bytes类型之间的大小是不一致的  3.在两次send直接会出现粘包的问题(通过一次recv最来解决)

          粘包是因为两次send在缓存区,会一次性发送

server端

import os
import socket
server = socket.socket()
server.bind(("127.0.0.1",6969))
server.listen()
while True:
con,addr=server.accept()
print(addr)
while True:
try:
data=con.recv(1024)#在程序正常退出,也就是con的tcp连接正常中断,data直接为0执行下一步
except Exception as e:
break #防止在windows下,强制关闭,con通道错误而报错,linux下不用这样
if not data:break
cmd_line=str(data,"utf-8") cmd_result=os.popen(cmd_line,"r").read()
if not cmd_result: #当os.popen执行非系统命令时,返回值为空
cmd_result = "未知命令"
cmd_result=bytes(cmd_result,"utf-8")
cmd_result_len=bytes(str(len(cmd_result)),"utf-8")
#int与bytes之间不能直接转换。
#len 在str与bytes类型之间的长度是不一致的!!!
print("发送数据长度",str(cmd_result_len,"utf-8")) con.send(cmd_result_len)
flag=con.recv(1024) #通过一次recv来解决两个send之间粘包的问题。
con.send(cmd_result)

clinet端

import socket
client = socket.socket()
client.connect(('127.0.0.1',6969))
while True:
x=input(">>:")
if not x:continue
client.send(bytes(x,encoding="utf-8"))
recv_len=int(str(client.recv(1024),"utf-8"))
client.send(bytes("ok","utf-8"))
print(recv_len)
data=bytes()
while len(data) != recv_len:
data += client.recv(1024)
print(str(data,"utf-8"))

四、实现文件下载:

  在python中数据都是bytes类型,打开文件 以rb,wb的类型打开。

  1.在服务方:

    通过os模块,读取文件的大小,使用f.read(size)读取指定数量,通过该大小来循环发送数据

2.在客户端

    先接受数据的大小,通过已经接受的数据 与 文件大小来判断大小

服务器端

import os
import socket
server = socket.socket()
server.bind(("127.0.0.1",6969))
server.listen() while True:
con,addr = server.accept()
while True:
cmd=str(con.recv(1024),"utf-8")
print(cmd)
cmd = cmd.split()
if cmd[0] == "push":
print(cmd[0])
f = open(cmd[1], "rb")
file_size = os.stat(cmd[1]).st_size
print(file_size)
con.send(bytes(str(file_size), encoding="utf-8")) flag=con.recv(1024)
# for line in f:
# con.send(line)
data_size=0
while file_size != data_size:
data=f.read(1024)
con.send(data)
data_size += len(data)
print(data_size)
print("发送完毕") f.close()

客户端

import socket
import os
client = socket.socket()
client.connect(("127.0.0.1",6969)) while True:
cmd=bytes(input("->:"),"utf8")
client.send(cmd) file_size = int(str(client.recv(1024), "utf-8"))
client.send("ok".encode("utf-8"))
data_len = 0
f = open("1.new.avi","wb")
while file_size != data_len:
data = client.recv(1024)
data_len += len(data)
f.write(data)
print(data_len)
f.close()

 五、多client并发连接  sockserver

创建sockserver四大步骤:

  1. First, you must create a request handler class by subclassing the BaseRequestHandlerclass and overriding its handle() method; this method will process incoming requests.   
  2. Second, you must instantiate one of the server classes, passing it the server’s address and the request handler class.
  3. Then call the handle_request() orserve_forever() method of the server object to process one or many requests.
  4. Finally, call server_close() to close the socket.
import socketserver

class MyTCPHandler(socketserver.BaseRequestHandler):
"""
The request handler class for our server. It is instantiated once per connection to the server, and must
override the handle() method to implement communication to the
client.
""" def handle(self):
# self.request is the TCP socket connected to the client
self.data = self.request.recv(1024).strip()
print("{} wrote:".format(self.client_address[0]))
print(self.data)
# just send back the same data, but upper-cased
self.request.sendall(self.data.upper()) if __name__ == "__main__":
HOST, PORT = "localhost", 9999 # Create the server, binding to localhost on port 9999
server = socketserver.TCPServer((HOST, PORT), MyTCPHandler) # Activate the server; this will keep running until you
# interrupt the program with Ctrl-C
server.serve_forever()

这里还不能直接使用:

  我们调用多线程的类 就可以使用了

    server = socketserver.ThreadingTCPServer((HOST, PORT), MyTCPHandler)

python socket单线程通信的更多相关文章

  1. Python Socket单线程+阻塞模式

    Python之旅]第五篇(二):Python Socket单线程+阻塞模式 python Socket单线程 Socket阻塞模式 串行发送 摘要:  前面第五篇(一)中的一个Socket例子其实就是 ...

  2. Python socket (单线程)

    client, 客户端 code : 客户端主要方法, s.send(); s.sendall(); s.recv(); s.connect() class Client(object): def _ ...

  3. python - socket通信笔记

    参考: 通过编写聊天程序来熟悉python中多线程和socket的用法:https://www.cnblogs.com/mingjiatang/p/4905395.html python socket ...

  4. Python Socket通信原理

    [Python之旅]第五篇(一):Python Socket通信原理   python Socket 通信理论 socket例子 摘要:  只要和网络服务涉及的,就离不开Socket以及Socket编 ...

  5. Python网络编程03 /缓存区、基于TCP的socket循环通信、执行远程命令、socketserver通信

    Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命令.socketserver通信 目录 Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命 ...

  6. 第五十三节,socket模块介绍,socket单线程通讯

    socket单线程通讯,只能单线程通讯,不能并发 socket是基于(TCP.UDP.IP)的通讯.也叫做套接字 通讯过程由服务端的socket处理信息发送,由客户端的socket处理信息接收. so ...

  7. python socket编程 实现简单p2p聊天程序

    目标是写一个python的p2p聊天的项目,这里先说一下python socket的基础课程 一.Python Socket 基础课程 Socket就是套接字,作为BSD UNIX的进程通信机制,取后 ...

  8. 一小时学会用Python Socket 开发可并发的FTP服务器!!

    socket是什么 什么是socket所谓socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄.应用程序通常通过"套接字"向网络发出请求 ...

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

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

随机推荐

  1. JavaScript 模块化历程

    这是一篇关于js模块化历程的长长的流水账,记录js模块化思想的诞生与变迁,展望ES6模块化标准的未来.经历过这段历史的人或许会感到沧桑,没经历过的人也应该知道这段历史. 无模块时代 在ajax还未提出 ...

  2. HDU1159-Common Subsequence-LCS

    上次写题解写到一半,写的比较具体,没写完,忘记存草稿了...导致现在没心情了. Common Subsequence Time Limit: 2000/1000 MS (Java/Others)    ...

  3. find the nth digit(二分查找)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=1597 find the nth digit Time Limit: 1000/1000 MS (Jav ...

  4. python数据类型(二)

    一.List(列表) List(列表) 是 Python 中使用最频繁的数据类型. 列表可以完成大多数集合类的数据结构实现.列表中元素的类型可以不相同,它支持数字,字符串甚至可以包含列表(所谓嵌套). ...

  5. struts中用kindeditor实现的图片上传并且显示在页面上

    做公司网站的时候由于需要在内容属性中加入图片,所以就有了这个问题,本来一开始找几篇文章看都是讲修改kindeditor/jsp/file_manager_json.jsp和upload_json.js ...

  6. load和DOMContenLoaded的区别

    load和DOMContentLoaded的作用就是当页面加载完成的时候自动执行,但他们执行的时间点是不一样的. DOM文档加载步骤: (1)解析html结构 (2)加载外部脚本和样式表文件 (3)解 ...

  7. 属性getter和setter

    我们知道,对象属性是由名字.值和一组特性(attribute)构成的.在ECMAScript5 中,属性值可以用一个或两个方法替代,这两个方法就是getter和setter.由getter和sette ...

  8. 关于Struts传递json给easyui的随笔

    今天在公司写测试代码,由于公司用的是ssh框架做的商城项目,我想先实现下简单的增删改查,奈何没有很好的后台页面(毕竟不能测试代码直接在他的项目里改啊) 所以想到了淘淘商城中有这个后台的管理页面,打算一 ...

  9. 关于OELD屏显示电池电量的简易方法

    如何采集电源电压大家可能都熟悉,stm32的ADC+DMA能很方便迅速的帮我们采集到自己想要的电压数据.使用DMA进行数据搬运也能很好的减轻CPU的一部分压力.但是这样只是第一步--数据. 用户想看到 ...

  10. 关于将dede织梦data目录迁移出web目录

    关于将dede织梦data目录迁移出web目录织梦官方提供了一个教程,但是如果你是按照他们提供的教程做的话会出现很多问题.比如验证码问题,图片显示问题等等一大堆.织梦官方这种是很不负责任的,因为那个教 ...