• socket 网络编程


    本篇内容:

      1、socket

      2、socketserver

      3、IO多路复用

    socket起源于Unix,而Unix/Linux基本哲学之一就是“一切皆文件”,对于文件用【打开】【读写】【关闭】模式来操作。socket就是该模式的一个实现,socket即是一种特殊的文件,一些socket函数就是对其进行的操作(读/写IO、打开、关闭)

    Python 官方关于 Socket 的函数请看 http://docs.python.org/library/socket.html

    Socket Families(地址簇)

    socket.AF_UNIX unix本机进程间通信 

    socket.AF_INET IPV4 

    socket.AF_INET6  IPV6

    这些常量表示地址(和协议)家族,用于第一个参数套接字()。如果没有定义afunix常量,那么这个协议就不被支持。

    Socket Types

    socket.SOCK_STREAM  #for tcp

    socket.SOCK_DGRAM   #for udp 

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

    socket.SOCK_RDM  #是一种可靠的UDP形式,即保证交付数据报但不保证顺序。SOCK_RAM用来提供对原始协议的低级访问,在需要执行某些特殊操作时使用,如发送ICMP报文。SOCK_RAM通常仅限于高级用户或管理员运行的程序使用。

    socket方法:

    socket.socket(family=AF_INET, type=SOCK_STREAM, proto=0, fileno=None) 
        创建socket实例,family为地址簇,type为协议类型,
    sk.bind(address)
      s.bind(address) 将套接字绑定到地址。address地址的格式取决于地址族。在AF_INET下,以元组(host,port)的形式表示地址。
    sk.listen(backlog)
      开始监听传入连接。backlog指定在拒绝连接之前,可以挂起的最大连接数量。
          backlog等于5,表示内核已经接到了连接请求,但服务器还没有调用accept进行处理的连接个数最大为5
          这个值不能无限大,因为要在内核中维护连接队列
    sk.setblocking(bool)
      是否阻塞(默认True),如果设置False,那么accept和recv时一旦无数据,则报错。
    sk.accept()
      接受连接并返回(conn,address),其中conn是新的套接字对象,可以用来接收和发送数据。address是连接客户端的地址。
      接收TCP 客户的连接(阻塞式)等待连接的到来
    sk.connect(address)
      连接到address处的套接字。一般,address的格式为元组(hostname,port),如果连接出错,返回socket.error错误。
    sk.connect_ex(address)
      同上,只不过会有返回值,连接成功时返回 0 ,连接失败时候返回编码,例如:10061
    sk.close()
      关闭套接字
    sk.recv(bufsize[,flag])
      接受套接字的数据。数据以字符串形式返回,bufsize指定最多可以接收的数量。flag提供有关消息的其他信息,通常可以忽略。
    sk.recvfrom(bufsize[.flag])
      与recv()类似,但返回值是(data,address)。其中data是包含接收数据的字符串,address是发送数据的套接字地址。
    sk.send(string[,flag])
      将string中的数据发送到连接的套接字。返回值是要发送的字节数量,该数量可能小于string的字节大小。即:可能未将指定内容全部发送。
    sk.sendall(string[,flag])
      将string中的数据发送到连接的套接字,但在返回之前会尝试发送所有数据。成功返回None,失败则抛出异常。
          内部通过递归调用send,将所有内容发送出去。
    sk.sendto(string[,flag],address)
      将数据发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回值是发送的字节数。该函数主要用于UDP协议。
    sk.settimeout(timeout)
      设置套接字操作的超时期,timeout是一个浮点数,单位是秒。值为None表示没有超时期。一般,超时期应该在刚创建套接字时设置,因为它们可能用于连接的操作(如 client 连接最多等待5s )
    sk.getpeername()
      返回连接套接字的远程地址。返回值通常是元组(ipaddr,port)。
    sk.getsockname()
      返回套接字自己的地址。通常是一个元组(ipaddr,port)
    sk.fileno()
      套接字的文件描述符

    Socket server端

    import socket
    server = socket.socket()
    server.bind(("localhost",333))
    server.listen()
    while True:
        try:
            conn,addr = server.accept()
        except ConnectionResetError as e:
            print(e)
    #    server.setblocking(False)
        finally:
            while True:
                data = conn.recv(1024)
                if  not data:
                    break
                print(data.decode())
                conn.send(data)
    server.close()

    Socket cliend端

    import socket
    client = socket.socket()
    client.connect(("localhost",9998))
    while True:
        message = input(">>:").strip()
        if len(message) == 0:
            continue
        client.send(message.encode())
        data = client.recv(1024)
        print(data.decode())
    client.close()

    上面代码只能连接一个客户端,但当客户端断开时server端会死掉。

    socket server :

    创建socket server需要以下几步:

    1、首先,您必须通过子类化BaseRequestHandlerclass并覆盖它的handle()方法来创建一个请求处理程序类,该方法将处理传入的请求。

    2、其次,必须实例化一个服务器类,将其传递到服务器的地址和请求处理程序类。

    3、然后调用服务器对象的handle_request()或serve_forever()方法来处理一个或多个请求。

    4、最后,调用server_close()来关闭套接字。

    简单的socket server:

    import socketserver
    class MySocket_server(socketserver.BaseRequestHandler):
        def handle(self):
            while True:
                self.data = self.request.recv(1024).strip()
                print(self.client_address)
                print(self.data)
                self.request.sendall(self.data.upper())
    if __name__ == "__main__":
        HOST,PORT  = "localhost",8767
        server = socketserver.ThreadingTCPServer((HOST,PORT),MySocket_server)
        server.serve_forever()
    
    此时还不能实现多客户端连接,需要多客户端连接需要改为:
    server = socketserver.ThreadingTCPServer((HOST,PORT),MySocket_server)

    Socket server其他的方法:

    类socketserver。BaseServer(server_address RequestHandlerClass)
    这是模块中所有服务器对象的超类。它定义了下面的接口,但是没有实现大多数方法,这些方法都是在子类中完成的。这两个参数存储在各自的serveraddress和RequestHandlerClass属性中。
    fileno()
    返回服务器正在监听的套接字的整数文件描述符。该函数通常被传递给选择器,以便在同一进程中监视多个服务器。
    handle_request()
    处理一个请求。这个函数按顺序调用以下方法:getrequest()、verifyrequest()和processrequest()。如果处理程序类的用户提供的handle()方法引发了一个异常,那么将调用服务器的handleerror()方法。如果在超时秒内没有收到请求,将调用handle超时()和handlerequest()将返回。
    serve_forever(poll_interval = 0.5)
    处理请求直到显式关闭()请求。每隔一段时间就关闭一次。忽略了超时属性。它还调用serviceactions(),它可以由子类或mixin来提供特定于给定服务的操作。例如,ForkingMixIn类使用serviceactions()来清理僵尸进程。
    在3.3版中发生了更改:将serviceactions调用添加到serve永久性方法。
    service_actions()
    这在serve永存()循环中调用。这个方法可以被子类或mixin类覆盖,以执行特定于给定服务的操作,例如清理操作。
    新的3.3版本中。
    shutdown()
    告诉serve永久性()循环停止并等待它的执行。
    server_close()
    清理服务器。可能会被覆盖。
    address_family
    服务器套接字所属的协议的家族。常见的例子是套接字。AF_INET socket.AF_UNIX。
    RequestHandlerClass
    用户提供的请求处理程序类;为每个请求创建该类的实例。
    server_address
    服务器正在监听的地址。地址的格式会根据协议的不同而有所不同;请参阅套接字模块的文档以获得详细信息。对于Internet协议,这是一个元组,其中包含一个给出地址的字符串,以及一个整数端口号:(“127.0.0.1”,80)。
    套接字
    服务器将侦听传入请求的套接字对象。
    服务器类支持以下类变量:
    allow_reuse_address
    服务器是否允许重用一个地址。这个默认值为False,可以在子类中设置更改策略。
    request_queue_size
    请求队列的大小。如果处理单个请求需要很长时间,那么在服务器繁忙时到达的任何请求都会被放到队列中,直到请求队列大小的请求。一旦队列满了,来自客户机的进一步请求将得到一个“连接拒绝”错误。默认值通常为5,但这可以被子类覆盖。
    socket_type
    服务器使用的套接字类型;套接字。SOCK_STREAM套接字。sockdgram是两个常见的值。
    超时
    超时持续时间,以秒为单位,如果不需要超时,则为零。如果handlerequest()在超时期间没有收到传入的请求,则调用handle超时()方法。
    有各种各样的服务器方法可以被像TCPServer这样的基服务器类的子类覆盖;这些方法对服务器对象的外部用户没有用处。
    finish_request()
    实际上,通过实例化RequestHandlerClass和调用它的handle()方法来处理请求。
    get_request()
    必须接受来自套接字的请求,并返回包含新的套接字对象的2元组,用于与客户端和客户端的地址进行通信。
    client_address handle_error(请求)
    如果RequestHandlerClass实例的handle()方法引发一个异常,就会调用该函数。默认的操作是将traceback打印到标准输出,并继续处理进一步的请求。
    handle_timeout()
    当超时属性被设置为一个值而不是None时,这个函数将被调用,并且超时时间已经过去,没有收到任何请求。forking服务器的默认操作是收集已经退出的任何子进程的状态,而在线程服务器中这个方法什么都不做。
    client_address process_request(请求)
    调用finish()来创建RequestHandlerClass的一个实例。如果需要,这个函数可以创建一个新的进程或线程来处理请求;ForkingMixIn和ThreadingMixIn类会这样做。
    server_activate()
    由服务器的构造函数调用来激活服务器。TCP服务器的默认行为仅在服务器的套接字上调用监听()。可能会被覆盖。
    server_bind()
    由服务器的构造函数调用,将套接字绑定到所需的地址。可能会被覆盖。
    client_address verify_request(请求)
    必须返回一个布尔值;如果值为True,

     通过I/O多路复用实现socket多并发:

    selectors模块:

    该模块允许基于select模块原语构建的高级别和高效的/输出多路复用。鼓励用户使用这个模块,除非他们希望对使用的os级别原语进行精确控制。

    import selectors
    import socket
    sel = selectors.DefaultSelector()  #根据平台选择最佳的IO多路机制,比如linux就会选择epoll
    def accept(sock,mask):
        conn,addr = sock.accept()
        print('accepted',conn,'ip',addr)
        sel.register(conn,selectors.EVENT_READ,read)
    def read(conn,mask):
        try:
            data = conn.recv(1024)
        except ConnectionResetError as e:      #客户端断开后,服务端会死掉,抓住这个异常。
            print(e)
            print('closing',conn)
            sel.unregister(conn)
            conn.close
        else:
            print('echoing',repr(data),'to',conn)
            conn.send(data)
    sock = socket.socket()
    sock.bind(('localhost',2222))
    sock.listen()
    sock.setblocking(False)
    sel.register(sock,selectors.EVENT_READ,accept) #注册功能
    while True:
        events = sel.select()  #[(sock),(),()]   监听
        # print(key.data)       #accept   找出有活动的绑定函数
        # print(key.fileobj)    #sock     找出有活动的文件描述符
        for key,mask in events:
            callback = key.data
            callback(key.fileobj,mask)
  • 相关阅读:
    poj 2674 Linear world
    poj 3185 The Water Bowls
    The Largest Clique (uva11324)
    Proving Equivalences (LA 4287)
    强联通分量( HihoCoder 1185 )
    求点双联通分量(HihoCoder
    求桥,割点(HihoCoder
    欧拉回路
    uva10054
    表达式树(公共表达式消除 uva 12219)
  • 原文地址:https://www.cnblogs.com/chimeiwangliang/p/7159063.html
Copyright © 2020-2023  润新知