Python系列的学习重点大概先写到这篇为止,其他基础的东西可以去翻翻官方文件,以后在实作上有遇到一些有趣的议题会再以文章来记录。


本篇主要介绍了类的定义与使用方法,只提及需要特别注意的重点,

枝微末节的就不提了,可以参考官方文件或以其他编程语言为参照,

基本上如果有其他编程语言的底子,Python的学习曲线还是相对较短的。


此次作为示例的代码是一个C/S架构的简易Echo Server,

服务端开启一个Accept线程监听外来连线,然后再针对已接受的外部连线开启一个子线程处理网路通信。

主线程根据键盘输入决定是否结束程序。


服务端: [ EchoServer.py ]

# -*- coding: UTF-8 -*-
# Description: It's a socket echo server.
#              Simply use working thread to process the communication from client.

import socket
import threading
import sys

HOST = ""
PORT = 5226
isRunning = 1
socketList = []


class ClientThread(threading.Thread):
    def __init__(self, ss, address):
        super().__init__()
        self.ss = ss
        self.addr = address

    def run(self):
        while isRunning:
            try:
                data = self.ss.recv(1024)
            except ConnectionResetError:
                print("Connection is closed by remote client...")
                break
            except ConnectionAbortedError:
                print("slave socket is closed.")
                break
            if (not data) or (data == "exit"):
                print("Client thread exit from", self.addr)
                break
            self.ss.send(data)
        self.ss.close()
        for s in socketList:  # remove client socket from socketList
            if s == self.ss:
                socketList.remove(s)
                break


class AcceptThread(threading.Thread):
    def __init__(self, s):
        threading.Thread.__init__(self)
        self.s = s

    def run(self):
        while isRunning:
            try:
                conn, addr = self.s.accept()
            except:
                print("Master socket is closed.")
                break
            print("Connected by", addr)
            socketList.append(conn)  # add client socket into socketList
            client_thread = ClientThread(conn, addr)
            client_thread.start()


def main():
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_STREAM, 0)
    except socket.error as msg:
        print(msg)
        print("Could not open socket!")
        sys.exit(1)
    try:
        s.bind((HOST, PORT))
        s.listen(5)
    except socket.error as msg:
        s.close()
        print(msg)
        print("Could not open socket!")
        sys.exit(1)
    if s is None:
        print("Could not open socket!")
        sys.exit(1)

    print("Server is listening on %d" % PORT)
    socketList.append(s)  # add master socket into socket list

    accept_thread = AcceptThread(s)
    accept_thread.start()
    while input("enter 'exit' to terminate program:") != "exit":  # wait for user input "exit"
        pass
    global isRunning
    isRunning = 0  # Set all thread being stop
    for sock in socketList:  # Close all socket
        sock.close()
    print("Exit server program.")

if __name__ == "__main__":
    try:
        main()
    except KeyboardInterrupt:
        sys.exit(2)
    except Exception as e:
        print(e)




重点1

上述代码先以threading这个模组实现了线程类,如第15行和第42行的类定义:

ClientThread类和AcceptThread类继承了threading.Thread类,並且以__init__()函数定义了子类的constructor

这里需要注意的是第17行和第44行,这两行代码都是在呼叫父类的constructor,只不过第44行是旧版Python的写法,第17行是Python 3.0之后的写法。


第17行使用super()呼叫避免直接使用父类名称,关于super()这个概念可以参考Java。

第17行的新版写法同时还可以改写成以下的旧版写法:

super(ClientThread, self).__init__()

相形之下就显得啰嗦。


------


重点2

线程类中必须实现一个run()函数,以供线程启动后运行,如第21行、第47行。


------


重点3

main()函数中绑定socket、监听socket我就不多解释了,这些用法可以参考C/C++的socket编程。

重点在第82、83行:

第82行宣告了一个AcceptThread类的对象实例,并且将本地的master socket作为参数传入类的建构函数中,

再赋值给类的成员变量self.s,作为接受外部连线之用。

第83行用AcceptThread类的对象实例呼叫了start()函数启动了AcceptThread线程。



重点大概就是这些了,其它的枝微末节可以边看代码边理解,很简单的。

最后附上客户端代码。


------


客户端: [ EchoClient.py ]

# -*- coding: UTF-8 -*-
# Description: It's a socket echo client

import socket
import sys

HOST = "your.domain.net"
PORT = 5226
s = None

for res in socket.getaddrinfo(HOST, PORT, socket.AF_UNSPEC, socket.SOCK_STREAM, 0, socket.AI_PASSIVE):
    af, socktype, proto, canonname, sa = res
    try:
        s = socket.socket(af, socktype, proto)
    except socket.error as msg:
        print(msg)
        s = None
        continue
    try:
        s.connect(sa)
    except socket.error as msg:
        print(msg)
        s.close()
        s = None
        continue
    break
if s is None:
    print("Could not open socket!")
    sys.exit(1)

while 1:
    data = ""
    while data == "":
        data = input("Input math algorithm:")
    if data == "exit":
        break
    try:
        s.send(data.encode())
        data = s.recv(1024)
    except:
        print("Connection is closed by remote server...")
        break
    print("recv:%s" % data.decode())

s.close()
print("Exit client program.")




好了,今天就写到这边,我肚子疼,先屎遁去洗手间。