一、socket模块简介

Python 提供了两个级别访问的网络服务,低级别的网络服务支持基本的 Socket,提供了标准的BSD Sockets API,可以访问底层操作系统Socket接口的全部方法;高级别的网络服务模块 SocketServer, 提供了服务器中心类,可以简化网络服务器的开发。

socket不支持多并发,socketserver是对socket的再封装,简化网络服务器版的开发。

二、socket模块接口

1、socket类型

Python 中,用使用socket函数来创建套接字,语法格式如下:

sock = socket.socket([family[, type]])

family:

socket.AF_UNIX :只能够用于单一的Unix系统进程间通信

socket.AF_INET :指定使用IPv4协议进行服务器间网络通信

socket.AF_INET6:指定使用IPv6协议进行服务器间网络通信

type:

socket.SOCK_STREAM:TCP流式连接

socket.SOCK_DGRAM:UDP数据报文

socket.SOCK_RAW:原始套接字,普通的套接字无法处理ICMP、IGMP等网络报文,而SOCK_RAW可以;SOCK_RAW也可以处理特殊的IPv4报文;利用原始套接字,可以通过IP_HDRINCL套接字选项由用户构造IP头。

socket.SOCK_SEQPACKET:可靠的连续数据包服务

2、服务端接口

socket.bind( address )

绑定地址address 到套接字,address是一个元组(host,port),host代表主机,port代表端口号。

socket.listen(backlog):开启TCP监听。backlog指定在拒绝连接前,操作系统可以挂起的最大连接数量,至少为1,大部分应用程序设为5。

connection, address = socket.accept()

被动接受TCP客户端连接,(阻塞式)等待连接。调用accept()方法后,socket会进入waiting状态。客户请求连接时,accept()方法会建立连接并返回服务器。accept()返回一个含有两个元素的元组(connection,address)。connection是新的socket对象,服务器必须通过connection与客户通信; address是客户端的Internet地址。

3、客户端接口

socket.connect(address):主动初始化TCP服务器连接,address为元组(hostname,port),如果连接出错,返回socket.error错误。

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

4、公共接口

buf = socket.recv(size)

接收TCP数据。参数size指定接收数据的缓冲区的大小,返回接收的数据。

socket.send(buf):发送TCP数据,将buf中的数据发送到连接的套接字。返回要发送的字节数量,可能小于buf的字节大小。

socket.sendall(buf):发送TCP数据。将buf中的数据发送到连接的套接字,但在返回前会尝试发送所有数据。成功返回None,失败则抛出异常。

data,addr = socket.recvfrom(bufsize):从套接字接收数据,但返回(data,address)。data是接收数据的缓冲区,address是发送数据的套接字地址。

socket.sendto(data, (addr, port)):将数据data发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回发送的字节数。

socket.close():关闭套接字。

socket.getpeername():返回连接套接字的远程地址。返回值是元组(ipaddr,port)。

socket.getsocketname():返回套接字自己的元组(ipaddr,port)

socket.setsockopt(level,optname,value):设置给定套接字选项的值。

socket.getsockopt(level,optname[.buflen]):返回套接字选项的值。

socket.settimeout(timeout):设置套接字操作的超时,timeout是一个浮点数,单位是秒。值为None表示没有超时。一般,超时应该在刚创建套接字时设置,因为socket可能用于连接的操作(如connect())。

socket.gettimeout():返回当前超时的值,单位是秒,如果没有设置超时,则返回None。

socket.fileno():返回套接字的文件描述符。

socket.setblocking(flag):如果flag为0,则将套接字设为非阻塞模式,否则将套接字设为阻塞模式(默认值)。非阻塞模式下,如果调用recv()没有发现任何数据,或send()调用无法立即发送数据,那么将引起socket.error异常。

socket.makefile():创建一个与套接字相关连的文件。

三、socket编程

1、TCP编程

大多数网络通信连接都是可靠的TCP连接。创建TCP连接时,主动发起连接的叫客户端,被动响应连接的叫服务器;连接成功后,通信双方都能以流的形式发送数据。

在Python中用TCP协议进行Socket编程十分简单,对于客户端,要主动连接服务器的IP和指定端口,对于服务器,要首先监听指定端口,然后,对每一个新的连接,创建一个线程或进程来处理。

使用 socket 模块的 socket 函数来创建一个 socket 对象。socket 对象可以通过调用其它函数来设置一个 socket 服务。

TCP服务端与客户端编程模型如下:

Python3快速入门(十)Python3网络编程_第1张图片

服务端编程模型如下:

1、调用socket函数创建一个TCP套接字,返回套接字sock。

2、调用bind将sock绑定到已知地址,通常为ip和port。

3、调用listen将sock设为监听模式,准备接收来自各客户端的连接请求。

4、调用accept等待接受客户端连接请求。

5、如果接收到客户端请求,则accept返回,得到新的连接套接字。

6、调用rev接收来自客户端的数据,调用send向客户端发送数据。

7、与客户端通信结束,服务器端可以调用close。

sock = socket.socket(AF.INET,sock.SOCK_STREAM)

在学习过程中有什么不懂得可以加我的
python学习交流扣扣qun,784-758-214
×××里有不错的学习视频教程、开发工具与电子书籍。
与你分享python企业当下人才需求及怎么从零基础学习好python,和学习什么内容
sock.bind((ip,port))

sock.listen(backlog)

while True:  # 不断接收新连接

    conn,addr = sock.accept()  # 阻塞

    while True:  # 接收连接,多次通信

        print("new conn",addr)

        data = conn.recv(1024)  #官方建议最大8192

        conn.send(data.upper())

        # recv 默认是阻塞的

        if not data :

            break  # 客户端一断开,conn.recv接收的是空数据

# 只能同时服务一个连接

sock.close()

TCP服务端实例:


import socket

import threading

import time

# 处理客户端,sock为socket,addr为客户端地址

def tcp_server(sock, addr):

    print("Accept new connection from %s:%s" % addr)

    sock.send(b"What's your name?")

    while True:

        data = sock.recv(1024)

        time.sleep(1)

        if not data or data.decode("utf-8") == "disconnect":

            break

        sock.send(('Hello, %s!' % data.decode('utf-8')).encode('utf-8'))

    sock.close()

    print('Connection from %s:%s closed.' % addr)

if __name__ == "__main__":

    # 创建基于IPV4和TCP的socket

    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # 绑定地址到socket

    sock.bind(("127.0.0.1", 3288))

    # 设置最大连接数,并开始监听

    sock.listen(10)

    print("TCP Server is running")

    print("Wait for new Connection")

    while True:

        # 接收TCP客户端连接,阻塞等待连接

        sock_fd, addr = sock.accept()

        # 开启新线程对TCP连接进行处理

        thread = threading.Thread(target=tcp_server, args=(sock_fd, addr))

        thread.start()

客户端编程模型如下:

1、创建一个socket套接字。

2、调用connect()函数将套接字连接到服务器。

3、调用send()函数向服务器发送数据,调用recv()函数接收来自服务器的数据。

4、与服务器的通信结束后,客户端程序可以调用close()函数关闭套接字。

TCP客户端实例:


import socket

if __name__ == "__main__":

    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    sock.connect(("127.0.0.1", 3288))

    print(sock.recv(1024).decode("utf-8"))

    # 持续与服务器交互:

    while True:

        # 获取用户输入:

        msg = input('Your input:')

        if not msg or msg == 'quit':

            break

        # 发送数据:

        sock.send(msg.encode('utf-8'))

        # 输出服务器返回的消息

        print('From server:', sock.recv(1024).decode('utf-8'))

    # 发送断开连接的指令

    sock.send(b'disconnect')

    # 套接字关闭

    sock.close()

2、UDP编程

UDP是面向无连接的协议。使用UDP协议时,不需要建立连接,只需要知道对方的ip和port,就可以直接发数据包,但数据包能否到达是无法确定的。

虽然用UDP传输数据不可靠,但优点是与TCP相比,速度快,对于不要求可靠到达的数据,可以使用UDP协议。

UDP服务端实例:

在学习过程中有什么不懂得可以加我的
python学习交流扣扣qun,784-758-214
×××里有不错的学习视频教程、开发工具与电子书籍。
与你分享python企业当下人才需求及怎么从零基础学习好python,和学习什么内容
import socket

if __name__ == "__main__":

    # 创建基于IPV4和TCP的socket

    sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

    # 绑定地址到socket

    sock.bind(("127.0.0.1", 3288))

    print("UDP Server is running")

    print("Wait for Message...")

    while True:

        # 接收数据,recvfrom()方法返回数据和客户端的地址与端口

        data, addr = sock.recvfrom(1024)

        print("Received from %s:%s" % addr)

        sock.sendto(b"Hello, %s!" % data,addr)

UDP客户端实例:


import socket

if __name__ == "__main__":

    sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

    while True:

        msg = input("Your input:")

        if not msg or msg == 'quit':

            break

        sock.sendto(msg.encode('utf-8'), ('127.0.0.1', 3288))

        # 输出服务器返回的消息:

        print('From server:', sock.recv(1024).decode('utf-8'))

    sock.close()