您的位置:首页 > 编程语言 > Python开发

Python学习笔记(3):类、网路​​编程

2016-03-16 02:59 597 查看
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.")


好了,今天就写到这边,我肚子疼,先屎遁去洗手间。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: