Python学习之socket网络编程

http://www.cnblogs.com/linhaifeng/articles/6129246.html

Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。

所以,我们无需深入理解tcp/udp协议,socket已经为我们封装好了,我们只需要遵循socket的规定去编程,写出的程序自然就是遵循tcp/udp标准的。(也有人将socket说成ip+port,ip是用来标识互联网中的一台主机的位置,而port是用来标识这台机器上的一个应用程序,ip地址是配置到网卡上的,而port是应用程序开启的,ip与port的绑定就标识了互联网中独一无二的一个应用程序而程序的pid是同一台机器上不同进程或者线程的标识

基于文件类型的套接字家族

套接字家族的名字:AF_UNIX

unix一切皆文件,基于文件的套接字调用的就是底层的文件系统来取数据,两个套接字进程运行在同一机器,可以通过访问同一个文件系统间接完成通信

基于网络类型的套接字家族

套接字家族的名字:AF_INET

(还有AF_INET6被用于ipv6,还有一些其他的地址家族,不过,他们要么是只用于某个平台,要么就是已经被废弃,或者是很少被使用,或者是根本没有实现,所有地址家族中,AF_INET是使用最广泛的一个,python支持很多种地址家族,但是由于我们只关心网络编程,所以大部分时候我么只使用AF_INET)

套接字工作流程

先从服务器端说起。服务器端先初始化Socket,然后与端口绑定(bind),对端口进行监听(listen),调用accept阻塞,等待客户端连接。在这时如果有个客户端初始化一个Socket,然后连接服务器(connect),如果连接成功,这时客户端与服务器端的连接就建立了。客户端发送数据请求,服务器端接收请求并处理请求,然后把回应数据发送给客户端,客户端读取数据,最后关闭连接,一次交互结束。

 1 import socket
 2 socket.socket(socket_family,socket_type,protocal=0)
 3 {socket_family 可以是 AF_UNIX 或 AF_INET。socket_type 可以是 SOCK_STREAM 或 SOCK_DGRAM。protocol 一般不填,默认值为 0。}
 4 
 5 获取tcp/ip套接字
 6 tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
 7 
 8 获取udp/ip套接字
 9 udpSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
10 
11 由于 socket 模块中有太多的属性。我们在这里破例使用了'from module import *'语句。使用 'from socket import *',我们就把 socket 模块里的所有属性都带到我们的命名空间里了,这样能大幅减短我们的代码。
12 例如tcpSock = socket(AF_INET, SOCK_STREAM)
socket()模块函数用法
 1 服务端套接字函数
 2 s.bind()    绑定(主机,端口号)到套接字
 3 s.listen()  开始TCP监听
 4 s.accept()  被动接受TCP客户的连接,(阻塞式)等待连接的到来
 5 
 6 客户端套接字函数
 7 s.connect()     主动初始化TCP服务器连接
 8 s.connect_ex()  connect()函数的扩展版本,出错时返回出错码,而不是抛出异常
 9 
10 公共用途的套接字函数
11 s.recv()            接收TCP数据
12 s.send()            发送TCP数据(send在待发送数据量大于己端缓存区剩余空间时,数据丢失,不会发完)
13 s.sendall()         发送完整的TCP数据(本质就是循环调用send,sendall在待发送数据量大于己端缓存区剩余空间时,数据不丢失,循环调用send直到发完)
14 s.recvfrom()        接收UDP数据
15 s.sendto()          发送UDP数据
16 s.getpeername()     连接到当前套接字的远端的地址
17 s.getsockname()     当前套接字的地址
18 s.getsockopt()      返回指定套接字的参数
19 s.setsockopt()      设置指定套接字的参数
20 s.close()           关闭套接字
21 
22 面向锁的套接字方法
23 s.setblocking()     设置套接字的阻塞与非阻塞模式
24 s.settimeout()      设置阻塞套接字操作的超时时间
25 s.gettimeout()      得到阻塞套接字操作的超时时间
26 
27 面向文件的套接字的函数
28 s.fileno()          套接字的文件描述符
29 s.makefile()        创建一个与该套接字相关的文件
套接字函数

基于TCP的套接字

tcp是基于链接的,必须先启动服务端,然后再启动客户端去链接服务端

 1 #tcp服务端
 2 ss = socket() #创建服务器套接字
 3 ss.bind()      #把地址绑定到套接字
 4 ss.listen()      #监听链接
 5 inf_loop:      #服务器无限循环
 6     cs = ss.accept() #接受客户端链接
 7     comm_loop:         #通讯循环
 8         cs.recv()/cs.send() #对话(接收与发送)
 9     cs.close()    #关闭客户端套接字
10 ss.close()        #关闭服务器套接字(可选)
11 #tcp客户端
12 cs = socket()    # 创建客户套接字
13 cs.connect()    # 尝试连接服务器
14 comm_loop:        # 通讯循环
15     cs.send()/cs.recv()    # 对话(发送/接收)
16 cs.close()            # 关闭客户套接字
基础demo
 1 #socket服务器端      (从客户端接收的信息并发送给客户端)
 2 import socket             #导入socket模块
 3 s = socket.socket()     #创建socket对象
 4 host = socket.gethostname()  #获取本地主机名
 5 port = 1234
 6 s.bind((host, port))          # 绑定端口号
 7 s.listen(5)                         # 设置最大连接数,超过后排队
 8 while True:   #循环连接
 9     c, addr = s.accept()           #建立客户端连接
10     print("连接地址是: %s" % str(addr))
11     m=c.recv(1024).decode()    # 接收小于 1024 字节的数据并解码
12     c.send(m.encode())
13     c.close()
14 s.close()
15 
16 #socket客户端
17 import socket
18 s = socket.socket()
19 host = socket.gethostname()
20 port = 1234
21 s.connect((host, port))
22 s.send('hello'.encode())
23 msg = s.recv(1024)
24 print(msg.decode())
简单demo
 1 #服务器端
 2 from socket import *
 3 ip_port=('127.0.0.1',8080)
 4 back_log=5
 5 buffer_size=1024
 6 tcp_server=socket(AF_INET,SOCK_STREAM)
 7 tcp_server.bind(ip_port)
 8 tcp_server.listen(back_log)
 9 
10 while True:
11     print('服务端开始运行了')
12     conn,addr=tcp_server.accept() #服务端阻塞
13     print('双向链接是',conn)
14     print('客户端地址',addr)
15 
16     while True:
17         # try:
18             data=conn.recv(buffer_size)
19             print('客户端发来的消息是',data.decode('utf-8'))
20             conn.send(data.upper())
21         # except Exception:
22         #     break
23     conn.close()
24 
25 tcp_server.close()
26 #客户端
27 from socket import *
28 ip_port=('127.0.0.1',8080)
29 back_log=5
30 buffer_size=1024
31 
32 tcp_client=socket(AF_INET,SOCK_STREAM)
33 tcp_client.connect(ip_port)
34 
35 while True:
36     msg=input('>>: ').strip()
37     if not msg:continue
38     tcp_client.send(msg.encode('utf-8'))
39     print('客户端已经发送消息')
40     data=tcp_client.recv(buffer_size)
41     print('收到服务端发来的消息',data.decode('utf-8'))
42 
43 tcp_client.close()
linhaifengDemo

打电话示例:

问题:

有的同学在重启服务端时可能会遇到

这个是由于你的服务端仍然存在四次挥手的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))
方法一
 1 发现系统存在大量TIME_WAIT状态的连接,通过调整linux内核参数解决,
 2 vi /etc/sysctl.conf
 3 
 4 编辑文件,加入以下内容:
 5 net.ipv4.tcp_syncookies = 1
 6 net.ipv4.tcp_tw_reuse = 1
 7 net.ipv4.tcp_tw_recycle = 1
 8 net.ipv4.tcp_fin_timeout = 30
 9  
10 然后执行 /sbin/sysctl -p 让参数生效。
11  
12 net.ipv4.tcp_syncookies = 1 表示开启SYN Cookies。当出现SYN等待队列溢出时,启用cookies来处理,可防范少量SYN攻击,默认为0,表示关闭;
13 
14 net.ipv4.tcp_tw_reuse = 1 表示开启重用。允许将TIME-WAIT sockets重新用于新的TCP连接,默认为0,表示关闭;
15 
16 net.ipv4.tcp_tw_recycle = 1 表示开启TCP连接中TIME-WAIT sockets的快速回收,默认为0,表示关闭。
17 
18 net.ipv4.tcp_fin_timeout 修改系統默认的 TIMEOUT 时间
方法二

基于UDP的套接字

udp是无链接的,先启动哪一端都不会报错

 1 UDP服务端:
 2 ss = socket()   #创建一个服务器的套接字
 3 ss.bind()       #绑定服务器套接字
 4 inf_loop:#服务器无限循环
 5     cs = ss.recvfrom()/ss.sendto() # 对话(接收与发送)
 6 ss.close()                         # 关闭服务器套接字
 7 UDP客户端:
 8 cs = socket()   # 创建客户套接字
 9 comm_loop:      # 通讯循环
10     cs.sendto()/cs.recvfrom()   # 对话(发送/接收)
11 cs.close()                      # 关闭客户套接字
udp服务端客户端
 1 服务端:
 2 import socket
 3 ip_port=('127.0.0.1',9000)
 4 BUFSIZE=1024
 5 udp_server_client=socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
 6 
 7 udp_server_client.bind(ip_port)
 8 
 9 while True:
10     msg,addr=udp_server_client.recvfrom(BUFSIZE)
11     print(msg,addr)
12 
13     udp_server_client.sendto(msg.upper(),addr)
14 客户端:
15 import socket
16 ip_port=('127.0.0.1',9000)
17 BUFSIZE=1024
18 udp_server_client=socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
19 
20 while True:
21     msg=input('>>: ').strip()
22     if not msg:continue
23 
24     udp_server_client.sendto(msg.encode('utf-8'),ip_port)
25 
26     back_msg,addr=udp_server_client.recvfrom(BUFSIZE)
27     print(back_msg.decode('utf-8'),addr)
示例一
 1 服务端:
 2 from socket import *
 3 ip_port=('127.0.0.1',8080)
 4 buffer_size=1024
 5 udp_server=socket(AF_INET,SOCK_DGRAM) #数据报
 6 udp_server.bind(ip_port)
 7 
 8 while True:
 9     data,addr=udp_server.recvfrom(buffer_size)
10     print(data)
11     udp_server.sendto(data.upper(),addr)
12 客户端:
13 from socket import *
14 ip_port=('127.0.0.1',8080)
15 buffer_size=1024
16 udp_client=socket(AF_INET,SOCK_DGRAM)  #SOCK_DGRAM数据报式的套接字
17 while True:
18     msg=input('>>: ').strip()
19     udp_client.sendto(msg.encode('utf-8'),ip_port)
20     data,addr=udp_client.recvfrom(buffer_size)
21     print(data.decode('utf-8'))
示例二
 1 ntp服务端:
 2 from socket import *
 3 from time import strftime
 4 
 5 ip_port=('127.0.0.1',9000)
 6 bufsize=1024
 7 
 8 tcp_server=socket(AF_INET,SOCK_DGRAM)
 9 tcp_server.bind(ip_port)
10 
11 while True:
12     msg,addr=tcp_server.recvfrom(bufsize)
13     print('===>',msg)
14     
15     if not msg:
16         time_fmt='%Y-%m-%d %X'
17     else:
18         time_fmt=msg.decode('utf-8')
19     back_msg=strftime(time_fmt)
20 
21     tcp_server.sendto(back_msg.encode('utf-8'),addr)
22 
23 tcp_server.close()
24 ntp客户端:
25 from socket import *
26 ip_port=('127.0.0.1',9000)
27 bufsize=1024
28 
29 tcp_client=socket(AF_INET,SOCK_DGRAM)
30 
31 
32 
33 while True:
34     msg=input('请输入时间格式(例%Y %m %d)>>: ').strip()
35     tcp_client.sendto(msg.encode('utf-8'),ip_port)
36 
37     data=tcp_client.recv(bufsize)
38 
39     print(data.decode('utf-8'))
40 
41 tcp_client.close()
时间服务器示例

举个栗子:

 1 服务器端:
 2 from socket import *
 3 import subprocess
 4 ip_port=('127.0.0.1',8080)
 5 back_log=5
 6 buffer_size=1024
 7 
 8 tcp_server=socket(AF_INET,SOCK_STREAM)
 9 tcp_server.bind(ip_port)
10 tcp_server.listen(back_log)
11 
12 while True:
13     conn,addr=tcp_server.accept()
14     print('新的客户端链接',addr)
15     while True:
16         #
17         try:
18             cmd=conn.recv(buffer_size)
19             if not cmd:break
20             print('收到客户端的命令',cmd)
21 
22             #执行命令,得到命令的运行结果cmd_res
23             res=subprocess.Popen(cmd.decode('utf-8'),shell=True,
24                                  stderr=subprocess.PIPE,
25                                  stdout=subprocess.PIPE,
26                                  stdin=subprocess.PIPE)
27             err=res.stderr.read()
28             if err:
29                 cmd_res=err
30             else:
31                 cmd_res=res.stdout.read()
32 
33             #
34             if not cmd_res:
35                 cmd_res='执行成功'.encode('gbk')
36             conn.send(cmd_res)
37         except Exception as e:
38             print(e)
39             break
40 客户端:
41 from socket import *
42 ip_port=('127.0.0.1',8080)
43 back_log=5
44 buffer_size=1024
45 tcp_client=socket(AF_INET,SOCK_STREAM)
46 tcp_client.connect(ip_port)
47 while True:
48     cmd=input('>>: ').strip()
49     if not cmd:continue
50     if cmd == 'quit':break
51     tcp_client.send(cmd.encode('utf-8'))
52     cmd_res=tcp_client.recv(buffer_size)
53     print('命令的执行结果是 ',cmd_res.decode('gbk'))
54 tcp_client.close()
基于tcp实现远程执行命令

结果如下:

 

posted @ 2018-08-10 14:39  一只小妖  阅读(116)  评论(0)    收藏  举报