Python socket服务

套接字(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())

原文地址:https://www.cnblogs.com/wztshine/p/12079510.html

时间: 2024-10-11 14:24:26

Python socket服务的相关文章

python socket模拟游戏开服

python socket服务端 #!/usr/bin/python #-*- coding: utf-8 -*- import socket import os,sys host='127.0.0.1' #监听的IP port=60000       #监听的端口 buffer=1024      #接受的字节数 sock=socket.socket(socket.AF_INET,socket.SOCK_STREAM)  #建立socket对象 sock.bind((host,port)) #

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

SocketServer包对socket包进行了包装(封装),使得创建socket服务非常简单. TCPServer+BaseRequestHandler 使用TCPServer和BaseRequestHandler编写socket服务的样例. #-*- coding:utf-8 -*- from SocketServer import TCPServer, BaseRequestHandler import traceback class MyBaseRequestHandlerr(BaseR

Python socket模块实现TCP服务端客户端

Python socket模块实现TCP服务端客户端 写了详细的注释,如果有哪一行不明白,可留言哦. 服务端脚本 # _*_ coding: utf-8 _*_ __author__ = 'xiaoke' __date__ = '2018/6/13 14:39' # 这个脚本创建一个TCP服务器,它接收来自客户端的消息,然后将消息加上时间戳前缀并返回客户端 import socket from time import ctime HOST = '' PORT = 21567 BUFSIZ = 4

带错误处理的python socket server服务范例的代码

下面的内容段是关于带错误处理的python socket server服务范例的内容,应该是对码农们有用途. import socket, traceback host = '' port = 51423 s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) s.bind((host, port)) s.listen(1) whil

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

目标:实现客户端与服务端的socket通信,消息传输. 客户端 客户端代码: from socket import socket,AF_INET,SOCK_STREAM #服务端的ip地址 server_ip = '127.0.0.1' #服务端socket绑定的端口号 server_port = 20000 if __name__ == '__main__': while True: str_msg = input("请输入要发送信息:") if str_msg != "&

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

上一篇 我们学习了简单的 Python TCP Socket 编程,通过分别写服务端和client的代码了解主要的 Python Socket 编程模型.本文再通过一个样例来加强一下对 Socket 编程的理解. 聊天室程序需求 我们要实现的是简单的聊天室的样例,就是同意多个人同一时候一起聊天.每一个人发送的消息全部人都能接收到,类似于 QQ 群的功能,而不是点对点的 QQ 好友之间的聊天.例如以下图: 图来自:http://www.ibm.com/developerworks/linux/tu

socket服务端和客户端

#!/usr/bin/env python#encoding: utf-8import socketdef handle_request(client): buf = client.recv(1024) client.send("HTTP/1.1 200 OK\r\n\r\n") client.send("Hello, World") def main(): sock = socket.socket(socket.AF_INET, socket.SOCK_STREA

【Python之旅】第五篇(一):Python Socket通信原理

只要和网络服务涉及的,就离不开Socket以及Socket编程,下面就说说Python Socket通信的基本原理. 1.Socket socket也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄.应用程序通常通过"套接字"向网络发出请求或者应答网络请求.可以列举中国移动或者是中国电信等的电话客服,当然,也可以看下面的图片来作形象的说明. socket起源于Unix,而Unix/Linux基本哲学之一就是:一切皆文件,即都可以用"打开open-

python/socket编程之粘包

python/socket编程之粘包 粘包: 只有TCP有尿包现象,UDP永远不会粘包. 首先需要掌握一个socket收发消息的原理 发送端可以是1k,1k的发送数据而接受端的应用程序可以2k,2k的提取数据,当然也有可能是3k或者多k提取数据,也就是说,应用程序是不可见的,因此TCP协议是面来那个流的协议,这也是容易出现粘包的原因而UDP是面向笑死的协议,每个UDP段都是一条消息,应用程序必须以消息为单位提取数据,不能一次提取任一字节的数据,这一点和TCP是很同的.怎样定义消息呢?认为对方一次