套接字(socket)是一个抽象层,应用程序可以通过它发送或接收数据,可对其进行像对文件一样的打开、读写和关闭等操作。

1. 实现客户端发送字符,服务器返回大写的字符:

服务器:

import socketserver

class MyTCPHandler(socketserver.BaseRequestHandler):         # 通过类的继承,实现
def handle(self): # 重写父类的handle方法,所有的操作都在此方法中
while True: # 循环,不停的接收从客户端来的数据
try:
self.data = self.request.recv(1024).strip() # 从客户端接收数据,每次收1024字节
print("{} send:".format(self.client_address), self.data)
self.request.send(self.data.upper()) # 从服务器发送给客户端数据
except ConnectionResetError as e:
print('Error: ',e)
break if __name__ == '__main__':
host,port = 'localhost',9999
server = socketserver.ThreadingTCPServer((host,port),MyTCPHandler) # 通过多线程实现多个客户端连接,每个客户端连接都是一个线程
server.serve_forever() # 一直运行服务

客户端:

import socket

client = socket.socket()                 # socket对象
client.connect(('localhost',9999)) # 连接服务器地址和端口 while True: # 循环,不停的输入发送数据
con = input('>>>:').strip()
if len(con) ==0: continue # 不能发送空数据,否则会阻塞
client.send(con.encode('utf-8')) # 发送数据,必须是二进制的
data = client.recv(1024) # 接收服务器返回的数据
print(data.decode()) # 打印 解码后的数据 client.close() # 关闭

2. 通过socket执行服务器命令:

用法:直接在客户端输入处输入命令如:ipconfig

服务器:

import socket
import os
import threading def tcplink(sock, addr):
print('Accept new connection from %s:%s...' % addr)
while True: # 和每个接入的客户端,进行多次数据通信
data = sock.recv(1024) # 接收客户端数据
if not data or data.decode('utf-8') == 'exit': # 如果客户端不发送数据或者发送了exit
print('client disconnected.')
break
content = os.popen(data.decode('utf-8')).read() # 对发送来的数据执行cmd命令,获取结果
if len(content) == 0: #如果执行的命令结果为空的,就手动造一个结果。因为如果为空数据,会挂起,无法正常发送。
content = 'cmd not exists.'
sock.send(str(len(content.encode('utf-8'))).encode('utf-8')) # 发送数据的长度
print('send length:', (len(content.encode('utf-8'))))
# print('content,', content.encode('utf-8'))
recv = sock.recv(1024) # 因为上下都有一个send连在一起,可能发生粘包现象,为了防止这种情况,可以让客户端重新应答一下
print('Answer:',recv.decode('utf-8'))
sock.send(content.encode('utf-8')) # 发送数据
print('send finished.')
sock.close()
print('Connection from %s:%s closed.' % addr) s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 监听端口:
s.bind(('127.0.0.1', 9999))
s.listen(3)
print('Waiting for connection...') while True:
# 接受一个新连接:
sock, addr = s.accept()
# 创建新线程来处理TCP连接:
t = threading.Thread(target=tcplink, args=(sock, addr))
t.start()

客户端:

import socket

# AF_INET 代表ipv4,SOCK_STREAM 代表TCP
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 确定网络协议,生成对象
s.connect(('127.0.0.1',9999)) # 连接服务器的地址和端口,元组的形式。
while True:
msg = input('>>:').strip()
if len(msg) != 0: # 如果消息为空,会一直挂起,所以不能为空
if msg =='exit':
s.close() # 关闭连接
print('Connection closed.')
break
s.send(msg.encode('utf-8')) # 给服务器发送数据,必须是二进制的
length = s.recv(1024) # 首先接收服务器返回的将要接收的数据的长度信息。
s.send(b'Ready to receive...') # 发送接收命令
length = int(length.decode('utf-8'))
print('receive len:', length)
data_len = 0
data_recv = b''
while data_len < length: # 已经接收的信息的长度,如果小于总长度
data = s.recv(1024) # 从服务器接收数据
data_recv += data
data_len += len(data)
print(data_recv.decode('utf-8')) # 打印返回的数据。

3. 通过socket传输文件:

用法:get 文件名

服务器:

import socket
import os
import hashlib
import threading def tcplink(sock, addr):
print('Accept new connection from %s:%s...' % addr)
while True: # 和每个接入的客户端,进行多次数据通信
data = sock.recv(1024) # 接收客户端数据
if not data or data.decode('utf-8') == 'exit': # 如果客户端不发送数据或者发送了exit
print('client disconnected.')
break
oper,filename = data.decode('utf-8').split() # 对接收的数据按照空格分割
if oper == 'get':
m = hashlib.md5()
if os.path.isfile(filename):
size = os.stat(filename).st_size # 获取文件大小
print('Send size:',size)
sock.send(str(size).encode('utf-8')) # 发送文件大小
recv = sock.recv(1024) # 接收客户端确认信息(因为上下文两个send是连着的,所以为了防止粘包,接收一次信息)
f = open(filename,'rb')
for line in f:
sock.send(line) #读取文件,发送给客户端
m.update(line)
# print('Send finished.',m.hexdigest()) # 打印md5的值
sock.send(m.hexdigest().encode('utf-8')) # 把md5的值发送给客户端
sock.close()
print('Connection from %s:%s closed.' % addr) s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 监听端口:
s.bind(('127.0.0.1', 9999))
s.listen(3)
print('Waiting for connection...') while True:
# 接受一个新连接:
sock, addr = s.accept()
# 创建新线程来处理TCP连接:
t = threading.Thread(target=tcplink, args=(sock, addr))
t.start()

客户端:

import socket
import hashlib # AF_INET 代表ipv4,SOCK_STREAM 代表TCP
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 确定网络协议,生成对象
s.connect(('127.0.0.1',9999)) # 连接服务器的地址和端口,元组的形式。
while True:
msg = input('>>:').strip()
if len(msg) != 0: # 如果消息为空,会一直挂起,所以不能为空
if msg =='exit':
s.close() # 关闭连接
print('Connection closed.')
break
s.send(msg.encode('utf-8')) # 给服务器发送数据,必须是二进制的
length = s.recv(1024) # 首先接收服务器返回的将要接收的数据的长度信息。
s.send(b'Ready to receive...') # 发送接收确认命令
length = int(length.decode('utf-8'))
print('Recv size:', length)
data_len = 0
data_recv = b''
# 新文件名
fileName = msg.split()[-1].split('.')[0]
fileExt = msg.split()[-1].split('.')[-1]
newFile = fileName+'-1.'+fileExt
f = open(newFile,'wb') # 打开文件,准备写入服务器发过来的文件
m = hashlib.md5()
while data_len < length: # 已经接收的信息的长度,如果小于总长度
size = length - data_len
if size > 1024: # 如果剩下的信息长度大于1024,即不能一次性发完。
size = 1024
else: # 如果能一次性发完,就只收剩下的信息。目的是准确的接收文件的大小,把可能粘连的send的数据留给下一次recv
size = length-data_len
data = s.recv(size) # 从服务器接收数据
f.write(data)
m.update(data)
data_len += len(data)
f.close()
print('recv_md5:',m.hexdigest()) # 打印返回的数据。
recv = s.recv(1024) # 接收下一次send的数据,即md5的值。
print('orig_md5:',recv.decode())

Python socket服务的更多相关文章

  1. 在python中编写socket服务端模块(二):使用poll或epoll

    在linux上编写socket服务端程序一般可以用select.poll.epoll三种方式,本文主要介绍使用poll和epoll编写socket服务端模块. 使用poll方式的服务器端程序代码: i ...

  2. Python使用TCPServer编写(多线程)Socket服务

    SocketServer包对socket包进行了包装(封装),使得创建socket服务非常简单. TCPServer+BaseRequestHandler 使用TCPServer和BaseReques ...

  3. Python socket编程客户端与服务端通信

    [本文出自天外归云的博客园] 目标:实现客户端与服务端的socket通信,消息传输. 客户端 客户端代码: from socket import socket,AF_INET,SOCK_STREAM ...

  4. Python网络编程基础 ❸ struct模块 基于upd的socket服务

    struct模块 基于upd的socket服务

  5. Python网络编程基础 ❷ 基于upd的socket服务 TCP黏包现象

    TCP的长连接 基于upd的socket服务 TCP黏包现象

  6. Java中使用Socket连接判断Inputstream结束,java tcp socket服务端,python tcp socket客户端

    最近在试着用java写一个socket的服务器,用python写一个socket的客户端来完成二者之间的通信,但是发现存在一个问题,服务器方面就卡在读取inputsream的地方不动了,导致后面的代码 ...

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

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

  8. Python Socket 网络编程

    Socket 是进程间通信的一种方式,它与其他进程间通信的一个主要不同是:它能实现不同主机间的进程间通信,我们网络上各种各样的服务大多都是基于 Socket 来完成通信的,例如我们每天浏览网页.QQ ...

  9. Python—Socket

    Socket模块 socket通常也称作"套接字",用于描述IP地址和端口,是特定网络协议如TCP/IP.UDP/IP套件对网络应用程序提供者提供的当前可移植标准的对象, 用来连接 ...

随机推荐

  1. python自然语言处理学习笔记2

    基础语法 搜索文本----词语索引使我们看到词的上下 text1.concordance("monstrous") 词出现在相似的上下文中 text1.similar(" ...

  2. Android AARCH64 平台的 OpenCL 配置

    原文地址:Android AARCH64 平台的 OpenCL 配置 Android AARCH64 平台的 OpenCL 配置 开发环境 IDE: Android Studio 3.4.1 Andr ...

  3. poi生成excel整理(设置边框/字体/颜色/加粗/居中/)

    转: poi生成excel整理(设置边框/字体/颜色/加粗/居中/) 2016年12月02日 11:05:23 吃奶的牛 阅读数:34324   HSSFWorkbook wb = new HSSFW ...

  4. delphi application 和 form 主窗体 都是窗口

    application     也是一个窗体,  和一般窗体不一样. 主窗体 是一个窗体. 然后这两个窗体,分别计算pid   获得结果竟然一样. 另外句柄是动态,全部都不一样.每一次都不一样.

  5. zabbix报警后不会自动消除解决

    http://www.cnblogs.com/zhongkai-27/p/9984597.html

  6. 【图形学手记】law of the unconscious statistician

    以扔色子为例,结果集为{1,2,3,4,5,6},每个数字出现的概率为1/6 以色子结果为随机变量X,如果我们定义函数F(X) = (X-3)2,我们来计算F(X)的概率分布: X=1,F(1)=(1 ...

  7. springmvc请求参数获取(自动绑定)的几种方法

    1.直接把表单的参数写在Controller相应的方法的形参中,适用于get方式提交,不适用于post方式提交. /** * 1.直接把表单的参数写在Controller相应的方法的形参中 * @pa ...

  8. Angular5 *ngIf 和 hidden 的区别

    问题 项目中遇到一个问题,有一个过滤查询的面板,需要通过一个展开折叠的button,来控制它的show 和 hide.这个面板中,有一个Select 组件,一个 input 查询输入框. 原来代码是: ...

  9. (转).net中的session与cookies区别及使用方法

    cookie数据存放在客户的浏览器上,session数据放在服务器上,cookie不是很安全,别人可以分析存放在本地的COOKIE并进行COOKIE欺骗,考虑到安全应当使用session   先介绍一 ...

  10. 第七周作业&实验报告5

     实验四 类的继承 实验目的 理解抽象类与接口的使用: 了解包的作用,掌握包的设计方法. 实验要求 掌握使用抽象类的方法. 掌握使用系统接口的技术和创建自定义接口的方法. 了解 Java 系统包的结 ...