本节导读:python
一 什么是scoketlinux
Socket是应用层与TCP/IP协议族通讯的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来讲,一组简单的接口就是所有。shell
socket起源于Unix,而Unix/Linux 基本哲学之一就是“一切皆文件”,均可以用“打开open –> 读写write/read –> 关闭close”模式 来操做。Socket就是该模式的一个实现,socket便是一种特殊的文件,一些socket函数就是对其进行的操做(读/写IO、打开、关闭)编程
你想给另外一台计算机发消息,你知道他的IP地址,他的机器上同时运行着qq、迅雷、word、浏览器等程序,你想给他的qq发消息,那想一下,你如今只能经过ip找到他的机器,但若是让这台机器知道把消息发给qq程序呢?答案就是经过port,一个机器上能够有0-65535个端口,你的程序想从网络上收发数据,就必须绑定一个端口,这样,远程发到这个端口上的数据,就全会转给这个程序设计模式
二 socket的工做流程浏览器
下面咱们举个打电话的小例子来讲明一下缓存
若是你要给你的一个朋友打电话,先拨号,朋友听到电话铃声后提起电话,这时你和你的朋友就创建起了链接,就能够讲话了。等交流结束,挂断电话结束这次交谈。 生活中的场景就解释了这工做原理。安全
(若是你去一家餐馆吃饭,假设哪里的老板就是服务端,而你本身就是客户端,当你去吃饭的时候,你确定的知道那个餐馆,也就是服务端的地址吧,可是对于你本身来讲,餐馆的老板不须要知道你的地址吧)服务器
三 socket套接字方法网络
socket参数
family(socket家族)
socket type类型
(Only SOCK_STREAM and SOCK_DGRAM appear to be generally useful.)
proto=0 请忽略,特殊用途
fileno=None 请忽略,特殊用途
socket函数
phone.bind('主机ip地址',端口号) #绑定到(主机,端口号)套接字 phone.listen() #开始TCP监听 phone.accept() #被动接受TCP客户的链接,等待链接的到来
s.connect() #主动初始化TCP服务器链接 s.connect_ex() #connect()函数的扩展版本,出错时返回出错码,而不是抛出异常
s.recv() #接收数据 s.send()# 发送数据(send在待发送数据量大于己端缓存区剩余空间时,数据丢失,不会发完,可后面经过实例解释) s.sendall() #发送完整的TCP数据(本质就是循环调用send,sendall在待发送数据量大于己端缓存区剩余空间时,数据不丢失,循环调用send直到发完) s.recvfrom() #Receive data from the socket. The return value is a pair (bytes, address) s.getpeername() #链接到当前套接字的远端的地址 s.sendto() #发送UDP数据 s.getsockname() #返回指定套接字的参数 s.setsockopt() #设置指定套接字的参数 s.close() # 关闭套接字 socket.setblocking(flag) #True or False,设置socket为非阻塞模式,之后讲io异步时会用 socket.getaddrinfo(host, port, family=0, type=0, proto=0, flags=0) #返回远程主机的地址信息,例子 socket.getaddrinfo('luffycity.com',80) socket.getfqdn()# 拿到本机的主机名 socket.gethostbyname() # 经过域名解析ip地址
phone.setblocking() #设置套接字的阻塞与非阻塞模式 phone.settimeout() #设置阻塞套接字操做的超时时间 phone.gettimeout() #获得阻塞套接字操做的超时时间
phone.fileno() # 套接字的文件描述符 phone.makefile() #建立一个与该套接字相关的文件
四 socket 服务端,客户端的建立
socket服务端建立
import socket phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)#买手机 phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1) #能够屡次启动 #执行屡次的时候会报错,那么怎么办呢、?就在绑卡前面加上上面那句setsockopt方法就ok了 phone.bind(('192.168.20.44',8080))#绑定手机卡(ip,端口) # 端口号在1024之前的是系统用的,1024之后的都是你本身写的程序去定义的端口 print('starting run......') phone.listen(5) #开机 5表明的是最多挂起5个,也能够好多个 while True: #连接循环 coon,client_addr=phone.accept()#等待接电话,(coon是创建的连接,客户端的ip和端口号组成的元组) print(coon,client_addr) #收发消息 while True: #通讯循环 try: #若是不加try...except ,就会报错,由于它不知道你何时断开连接的,服务器还觉得你在运行 data = coon.recv(1024) #收了1024个字节的消息 print('client data 收到消息:%s'%data.decode('utf-8')) coon.send(data.upper()) #发消息 except Exception: #由于你不知道客户端何时断开连接, break coon.close() #挂电话 phone.close() #关机 # 处理逻辑错误的两种方式: # if 判断 # try...except 异常处理 # 异常处理 # 当你知道直接错误的条件时就用if判断了 # 当程序错误必定发生,可是你又预知不了它出错的条件是什么的时候,就用try...except 服务端
socket客户端建立
import socket phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)#买手机 phone.connect(('192.168.20.44',8080)) #直接链接服务端的ip和端口 # 发收消息 while True: msg = input('>>:').strip() #用户输入 if not msg:continue #若是为空就继续输 phone.send(msg.encode('utf-8')) # 发送你输入的消息 # phone.send('hello'.encode('utf-8')) data = phone.recv(1024) #在接收一下 print('server back res服务端返回结果:>>%s'%data.decode('utf-8')) phone.close()
注意:
若是你在重启服务端的时候可能遇到这样的问题:
这个是因为你的服务端仍然存在四次挥手的time_wait状态在占用地址(若是不懂,请深刻研究1.tcp三次握手,四次挥手 2.syn洪水攻击 3.服务器高并发状况下会有大量的time_wait状态的优化方法)。那么怎么解决呢?你也能够这样的
1 #加入一条socket配置,重用ip和端口 2 3 phone=socket(AF_INET,SOCK_STREAM) 4 phone.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #就是它,在bind前加 5 phone.bind(('127.0.0.1',8080))
基于tcp协议模拟ssh远程执行命令
import socket import subprocess phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM)#买手机 phone.bind(('192.168.20.44',8081))#绑定手机卡 phone.listen(5)#阻塞的最大个数 print('starting....') while True: conn,addr=phone.accept()#等待链接 print(addr,conn) while True: cmd=conn.recv(10240)#接收的最大值 # if not cmd :break print('接收的是:%s'%cmd.decode('utf-8')) #处理过程 res=subprocess.Popen(cmd.decode('utf-8'),shell=True, #Popen是执行命令的方法 stdout=subprocess.PIPE, stderr=subprocess.PIPE ) stdout=res.stdout.read() stuerr=res.stderr.read() conn.send(stdout+stuerr) conn.close() phone.close()
基于udp协议的socket
from socket import * udp_server = socket(AF_INET,SOCK_DGRAM) udp_server.bind(('127.0.0.1',8080)) #绑定 while True:#通信循环 msg,client_addr= udp_server.recvfrom(1024) print('收到的消息是:%s'%msg.decode('utf-8')) udp_server.sendto(msg.upper(),client_addr) udp_server.close()