sock模块

python学习-sock模块

about

Python网络通讯主要是C/S架构的,采用套接字实现。C/S架构是客户端(Client)和服务端(Server)架构,Server唯一的目的就是等待Client的请求,Client连上Server发送必要的数据,然后等待Server端完成请求的反馈。

  1. C/S网络编程:
    Server端进行设置,首先创建一个通信端点,让Server端能够监听请求,之后就进入等待和处理Client请求的无限循环中。Client编程相对Server端编程简单,只要创建一个通信端点,建立到服务器的链接,就可以提出请求了。

  2. tcp 与 udp
    tcp:通信之前一定要建立一条连接,要创建TCP套接字就得创建时指定套接字类型为SOCK_STREAM。
    udp:无需建立连接就可以通讯,要创建UDP套接字就得创建时指定套接字类型为SOCK_DGRAM。

  3. 套接字
    套接字是一种具有之前所说的“通信端点”概念的计算网络数据结构,网络化的应用程序在开始任何通信都必须创建套接字。Python支持:AF_UNIX、AF_NETLINK、AF_INET,其中AF_INET是基于网络的套接字。

模块语法

# 使用socket模块的socket()函数来创建套接字
socket(socket_family, socket_type, protocol=0)
# socket_family:AF_UNIX、AF_INET
# socket_type可:SOCK_STREAM、SOCK_DGRAM
# protocol一般不填,默认值是0

# 创建一个TCP/IP套接字
tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 创建一个UDP/IP套接字
udpSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

from socket import *
# socket对象中有太多属性,可以大幅缩短代码,调用如下
tcpSock = socket(AF_INET, SOCK_STREAM)

套接字对象

服务器端套接字函数

# 绑定地址(主机号、端口号对)到套接字
s.bind()

# 开始 tcp 监听
s.listen()

# 被动接受 tcp 客户端连接,(阻塞式)等待连接的到来
s.accept()

客户端套接字函数

# 主动初始化 tcp 服务器连接
s.connect()

# connect() 函数扩展版本,出错时返回出错码而不是抛出异常
s.connect_ex()

公共用途的套接字函数

# 接受 tcp 数据
s.recv()

# 发送 tcp 数据
s.send()
# 返回发送的字节大小。这个字节长度可能少于实际要发送的数据的长度。换句话说,这个函数执行一次,并不一定能发送完给定的数据,可能需要重复多次才能发送完成。

# 完整发送 tcp 数据
s.sendall()
# 发送完整的TCP数据,成功返回None,失败抛出异常


# 接受 udp 数据
s.recvfrom()

# 发送 udp 数据
s.sendto()

# 连接到当前套接字的远端地址(tcp 连接)
s.getpeername()

# 获取当前套接字的地址
s.getsockname()

# 返回指定套接字的参数
s.getsockopt()

# 设置指定套接字的参数
s.setsockopt()

# 关闭套接字
s.close()

面向模块的套接字函数

# 设置套接字的阻塞与非阻塞模式
s.setblocking()

# 设置阻塞套接字操作的超时时间
s.settimeout()

# 得到阻塞套接字操作的超时时间
s.gettimeout()

面向文件的套接字函数

# 套接字的文件描述符
s.fileno()

# 创建一个与套接字关联的文件对象
s.makefile()

发送 http 及 https

# https协议
import socket
import ssl
sock = ssl.wrap_socket(socket.socket())
sock.connect((host, 443))
sock.send(data.encode())
//sock.sendall(data.encode())
data = sock.recv(BUFFSIZE).decode()
sock.close()

# http协议
import socket
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.connect((host, port))
sock.send(data.encode())
//sock.sendall(data.encode())
data = sock.recv(BUFFSIZE).decode()
sock.close()

# http代理
import socket
import socks  # pip install PySocks
socks.set_default_proxy(socks.HTTP,addr='127.0.0.1',port=80) #设置socks代理
socket.socket = socks.socksocket  # 把代理应用到socket
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.connect((host, port))

# 注意项
1.在GET请求末尾的地方,应该有两个 "\r\n,如果只有一个的话就会一直就返回错误400,第一个 "\r\n"是代表一行结束了第二个"\r\n"代表http数据包消息结束,\r : return 到当前行的最左边。 \n: newline 向下移动一行,并不移动左右。 Linux中\n表示回车+换行; Windows中\r\n表示回车+换行。
2.Connection如果不定义,默认是keep-alive,那么服务端在返回数据后不会断开连接,而是允许客户端继续使用这个连接发送请求

通信实例

注意
python3.5和Python2.7在套接字返回值解码上有区别:
python bytes和str两种类型可以通过函数encode()和decode()相互转换,
str→bytes:encode()方法。str通过encode()方法可以转换为bytes。
bytes→str:decode()方法。要把bytes变为str,就需要用decode()方法。
而如果我们从网络或磁盘上读取了字节流,那么读到的数据就是bytes。

服务器端

# tcpSerSock.py

from socket import *
from time import ctime

HOST = 'localhost'      # 主机号
PORT = 21567            # 端口号
BUFSIZE = 1024          # 缓冲区大小为 1k
ADDR = (HOST, PORT)

tcpSerSock = socket(AF_INET, SOCK_STREAM)
tcpSerSock.bind(ADDR)      # 绑定地址到套接字
tcpSerSock.listen(5)       # 监听,最多监听 5 个连接

while True:                # 无线循环等待连接
    try:
        print('Waiting for connection....')
        tcpCliSock, addr = tcpSerSock.accept()      # 被动接受客户端连接
        print('Connected client from: ', addr)

        while True:
            data = tcpCliSock.recv(BUFSIZE).decode()         # 接收数据
            if not data:
                  break
            else:
                  print('Client:', data)
            time_data = '[%s] %s'%(ctime(), data)
            tcpCliSock.send(time_data.encode())              # 发送时间戳
    except Exception as e:
        print('Error: ', e)

tcpSerSock.close()      # 关闭服务器
tcpCliSock.close()

客户端

# tcpCliSock.py

from socket import *

HOST = 'localhost'      # 服务器主机号
PORT = 21567            # 服务器端口号
BUFSIZE = 1024          # 缓冲区大小为 1k
ADDR = (HOST, PORT)

tcpCliSock = socket(AF_INET, SOCK_STREAM)
tcpCliSock.connect(ADDR)      # 连接服务器

while True:                   # 无限循环等待连接
    try:
        data = input('>').encode()
        if not data:
            break
        tcpCliSock.send(data)                           # 发送数据
        data = tcpCliSock.recv(BUFSIZE).decode()        # 接受数据
        if not data:
            break
        print('Server: ', data)
    except Exception as e:
        print('Error: ', e)

tcpCliSock.close()            # 关闭客户端

开始通信

# Server
'''
Waiting for connection....
Connected client from:  ('127.0.0.1', 51744)
Client: hello
Client: i am shivers
Waiting for connection....
'''

# Client
'''
>hello
Server:  [Sun Feb 14 16:25:23 2021] hello
>i am shivers
Server:  [Sun Feb 14 16:25:30 2021] i am shivers
>
'''

https://mp.weixin.qq.com/s?__biz=MzU2MTQwMzMxNA==&mid=2247488767&idx=2&sn=024a7173a1250bec186d7e7b70be504c&chksm=fc781022cb0f9934144d188faa2e96943011d2e4816352aa48149f31bbbffe891fb07e501de1&scene=21#wechat_redirect

posted @ 2021-02-14 16:31  Shivers0x72  阅读(154)  评论(0编辑  收藏  举报