设为首页 收藏本站
查看: 1728|回复: 0

[经验分享] python中socket与socketserver模块简单入门使用

[复制链接]

尚未签到

发表于 2018-8-8 12:13:34 | 显示全部楼层 |阅读模式
  一、socket模块
  socket又叫套接字,是网络编程中的一个基本组件,是两个端点的程序之间的“信息通道”程序可分布在不同的计算机上(通过网络连接),通过socket套接字相互发送信息。python中的大多数的网络编程都 隐藏了socket模块的基本细节。
  python中通过socket模块完成网络编程的套接字实现,一个套接字就是socket模块中的socket类的一个实例。socket实例化需要三个参数分别是family(ipv4,ipv6,unix)其中默认是ipv4 "socket.AF_INET",第二个参数是流,默认是socket.SOC_STREAM表示tcp,或socket.SOCK_DGRAM,第三个参数是协议,默认是0,使用默认即可。因此实际上实例化出一个套接字,只需要二个参数。
  socket通讯原理:
DSC0000.jpg

  socket函数
  python 3.x下发送的内容必须是byte类型,2.x是字符串
DSC0001.jpg

DSC0002.jpg

  二、socket简单通讯
  环境说明:ubuntu 16.04    python 3.5.2    模拟的服务器和客户端均在一台机器上(可以在不同机器上,只要保障网络连接)
  先来看一个简的服务器与客户端的socket通讯
  功能:服务器侦听等待连接,客户端连接后发送连接信息,之后客户端发送的任何信息,服务器端原样返回客户端。
  代码如下:
  1、srv.py
  

import socket  
sk = socket.socket()
  
#print(sk)
  
HOST = '127.0.0.1'
  
PORT = 2222
  
BUFSIZ = 1024
  
ADDR = (HOST,PORT)
  

  
sk.bind(ADDR)
  
sk.listen(5)
  
print("waiting...")
  

  
while True:
  conn,addr = sk.accept()
  ip,port = addr
  print("Got connection from",addr)
  conn.send(bytes("connected from %s:%s." %addr,"utf-8"))
  while True:
  data = conn.recv(BUFSIZ)
  if not data:
  break                                     #直接回车退出本次连接
  print(data)
  conn.send(data)
  

  
conn.close()
  

  cli.py
  

import socket  
HOST = '127.0.0.1'
  
PORT = 2222
  
BUFSIZ = 1024
  
ADDR = (HOST,PORT)
  
sk = socket.socket()
  
sk.connect(ADDR)
  
print(sk.recv(BUFSIZ))
  
while True:
  inp = input(">>>")
  if not inp:                                       #忽略空格回车
  continue
  sk.send(bytes(inp,"utf-8"))
  print(sk.recv(BUFSIZ))
  

  

  运行结果:
DSC0003.jpg

DSC0004.jpg

  以上是一个最简单的socket网络通讯。以上可以发现双方通讯时一发一收,或一收一发,总是成对出现,但有一个缺点,就是同时只能给一个客户端提供服务(通讯)其他的客户端只能等待。后面通过socketserver来解决一服务器多客户端同时连接的情况。
  三、socket上传文件
  在完成基本的通讯后,我们来做一个文件上传的实例。
  server端
  

#coding:utf-8  
import socket
  
import os
  
sk = socket.socket()
  
#print(sk)
  
HOST = '127.0.0.1'            #指定ip
  
PORT = 2222                     #侦听端口
  
BUFSIZ = 1024                  #缓存大小
  
ADDR = (HOST,PORT)
  

  
sk.bind(ADDR)                   #套接字ip端 口进行绑定
  
sk.listen(5)                           #连接等待数,默认5
  
print("waiting....")
  
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
  

  
while True:
  conn, addr = sk.accept()
  print(addr)
  while True:
  data = conn.recv(1024)
  cmd,filename,filesize = (str(data,"utf-8")).split("|")
  path = os.path.join(BASE_DIR,'yuan',filename)
  filesize = int(filesize)
  

  f = open(path,"ab")
  has_receive = 0
  while has_receive != filesize:
  data = conn.recv(1024)
  f.write(data)
  has_receive+=len(data)
  f.close()
  

  
sk.close()
  

  client端
  

#coding:utf-8  
import socket
  
import os
  

  
HOST = '127.0.0.1'
  
PORT = 2222
  
BUFSIZ = 1024
  
ADDR = (HOST,PORT)
  
sk = socket.socket()
  
sk.connect(ADDR)
  

  
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
  
while True:
  inp = input(">>>").strip() #post|12.png
  cmd,path = inp.split("|")
  path = os.path.join(BASE_DIR,path)  ##完整文件路径xxx/12.png
  filename = os.path.basename(path)    ##获取文件名
  file_size = os.stat(path).st_size    #获取文件大小
  file_info = "post|%s|%s" %(filename,file_size)
  sk.sendall(bytes(file_info,"utf-8"))
  f = open(path,'rb')
  has_sent = 0
  while  has_sent != file_size:
  data = f.read(BUFSIZ)
  sk.sendall(data)
  has_sent+=len(data)
  f.close()
  print("上传成功~")
  
sk.close()
  

  总结:socket通讯的关键是一发一收,无论是服务端先发还是客户端先发,必须是一端发送另一端接收,如果连续发送会出现粘包现象。
  四、socketserver实例多并发通讯
  server.py
  

#coding:utf-8  
import socketserver    #能实现并发
  
#并发聊天
  
class MyServer(socketserver.BaseRequestHandler):
  print("服务端启动...")
  def handle(self):
  while True:
  conn = self.request
  print (self.client_address)
  while True:
  client_data=conn.recv(1024)
  print (str(client_data,"utf8"))
  print ("waiting...")
  server_response=input(">>>")
  conn.sendall(bytes(server_response,"utf8"))
  conn.close()
  

  
if __name__ == '__main__':
  server = socketserver.ThreadingTCPServer(('127.0.0.1',2223),MyServer)
  server.serve_forever()
  

  

  client.py
  

#coding:utf-8  
import socket
  

  
ip_port = ('127.0.0.1',2223)
  
sk = socket.socket()
  
sk.connect(ip_port)
  
print ("客户端启动:")
  
while True:
  inp = input('>>>')
  sk.sendall(bytes(inp,"utf-8"))
  if inp == 'exit':
  break
  server_response=sk.recv(1024)
  print (str(server_response,"utf-8"))
  
sk.close()
  

  执行结果服务器端通过多线程完成并发多客户端“同时”聊天效果。
  以上为socket socketserver基本应用。如有不当之处欢迎指正。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.iyunv.com/thread-548656-1-1.html 上篇帖子: Python19 内置函数 下篇帖子: python之commands模块(执行Linux Shell命令)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表