一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

腳本之家,腳本語言編程技術及教程分享平臺!
分類導航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服務器之家 - 腳本之家 - Python - Python的Socket編程過程中實現(xiàn)UDP端口復用的實例分享

Python的Socket編程過程中實現(xiàn)UDP端口復用的實例分享

2020-08-17 11:14liweisnake Python

這篇文章主要介紹了Python的Socket編程過程中實現(xiàn)UDP端口復用的實例分享,文中作者用到了Python的twisted異步框架,需要的朋友可以參考下

關于端口復用

一個套接字不能同時綁定多個端口,如果客戶端想綁定端口號,一定要調用發(fā)送信息函數(shù)之前綁定( bind )端口,因為在發(fā)送信息函數(shù)( sendto, 或 write ),系統(tǒng)會自動給當前網(wǎng)絡程序分配一個隨機端口號,這相當于隨機綁定了一個端口號,這里只會分配一次,以后通信就以這個隨機端口通信,我們再綁定端口號的話,就會綁定失敗。如果我們放在發(fā)送信息函數(shù)( sendto, 或 write )之前綁定,那樣程序將以我們綁定的端口號發(fā)送信息,不會再隨機分配一個端口號。實際上,默認的情況下,如果一個網(wǎng)絡應用程序的一個套接字 綁定了一個端口,這時候,別的套接字就無法使用這個端口。那如何讓兩個套接字都能成功綁定一個端口呢?這時候就需要要到端口復用了。端口復用允許在一個應用程序可以把 n 個套接字綁在一個端口上而不出錯。
端口復用能在系統(tǒng)已開放的端口上進行通訊,只對輸入的信息進行字符匹配,不對網(wǎng)絡數(shù)據(jù)進行任何攔截、復制類操作,所以對網(wǎng)絡數(shù)據(jù)的傳輸性能絲毫不受影響。
但要注意,建立連接后服務端程序占用極少系統(tǒng)資源,被控端不會在系統(tǒng)性能上有任何察覺,通常被后門木馬所利用。
在winsock的實現(xiàn)中,對于服務器的綁定是可以多重綁定的,在確定多重綁定使用誰的時候,根據(jù)一條原則是誰的指定最明確則將包遞交給誰,而且沒有權限之分,也就是說低級權限的用戶是可以重綁定在高級權限如服務啟動的端口上的,這是非常重大的一個安全隱患。

Python解決UDP端口復用問題
一直覺得UDP協(xié)議很簡單,但是今天問題讓我感覺到網(wǎng)絡的基礎真是博大精深。

    廢話少說,來看問題吧。由于協(xié)議的需要,我得實現(xiàn)一個UDP的客戶端和服務器端,并且從同一個端口讀寫數(shù)據(jù)。

    最初不以為然,無非就是用兩個socket,一個監(jiān)聽并從這個端口讀取數(shù)據(jù)(服務器端采用了twisted),另一個向這個端口寫入數(shù)據(jù),用python實現(xiàn)只要10行左右的代碼。

?
1
2
3
4
5
6
7
8
9
def startServer(queue, port):
  reactor.listenUDP(port, DhtResponseHandler(queue))
  reactor.run()
 
def sendUdpMsg(self, addr, msg):
  socketHandler = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
  socketHandler.bind(("", self.port))
  socketHandler.sendto(msg, addr)
  socketHandler.close()

     由于要向同一個端口寫數(shù)據(jù),于是client必須有bind,但是運行后發(fā)現(xiàn)server先bind了這個端口,client運行時會報錯

 

復制代碼 代碼如下:

 

error: [Errno 10048] Only one usage of each socket address (protocol/network address/port) is normally permitted  

 


     一般這種錯誤時因為多個socket不能同時bind同一個地址
     由于基礎不夠扎實,我開始瘋狂的搜索,發(fā)現(xiàn)有人說端口復用的問題,所謂的端口復用,是指一個套接字釋放掉一個端口后有一個wait_time,另一個套接字如果接著bind就會報錯。雖然我的問題不完全一樣,但是我欣喜若狂的使用了。即在client bind前加上如下一句

 

?
1
socketHandler.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)

     但是仍然報錯:

 

復制代碼 代碼如下:

 

 

error: [Errno 10013] An attempt was made to access a socket in a way forbidden by its access permissions  

 


    (順便一提,還有另一個參數(shù)叫SO_REUSEPORT,即復用端口,另外有一個叫SO_EXCLUSIVEADDRUSE,即不準復用該端口,其他socket的參數(shù)還有很多,可以參考winsockhttp://msdn.microsoft.com/en-us/library/aa924071.aspx或者unix下的socket)
     這個10013錯誤讓我百思不得其解,搜索一下,主要有兩種解釋,有人說是需要提升應用程序的權限為管理員,我用的是eclipse+pydev,提升完eclipse權限沒用,實際上還要修改python.exe的權限,方法是在這個程序上右鍵,兼容性一欄中勾上以系統(tǒng)管理員身份運行;有人說是跟其他程序地址或者端口沖突。但是我測試過發(fā)現(xiàn)都不行。

 

     另外,運行的時候發(fā)現(xiàn),twisted的服務器端一定是要在主線程中,否則會報signal一定要在主線程才能接受的錯誤,但是twisted的reactor一運行起來就阻塞了。

     在twisted文檔中翻到,原來還有一種UDP叫做connected UDP,變態(tài)吧,所謂connected UDP,就是只能向一個地址收發(fā)數(shù)據(jù),看起來貌似可以,但是不符合可以向多個地址接收數(shù)據(jù)。

     最后在一篇文章中翻到說需要兩個端口都設置重用,于是我試著重新寫一個服務器,與之前的客戶端配合,運行良好,完全無錯

?
1
2
3
4
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(("", port))
data, address = sock.recvfrom(4096)

     好吧,看來問題在調用twisted了,不知道他是否有這樣的設置,進去將這部分代碼翻了一下,找不到這樣設置的參數(shù)。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
class Port(abstract.FileHandle):
  def __init__(self, port, proto, interface='', maxPacketSize=8192,
         reactor=None):
    """
    Initialize with a numeric port to listen on.
    """
    self.port = port
    self.protocol = proto
    self.readBufferSize = maxPacketSize
    self.interface = interface
    self.setLogStr()
    self._connectedAddr = None
 
    abstract.FileHandle.__init__(self, reactor)
 
    skt = socket.socket(self.addressFamily, self.socketType)
    addrLen = _iocp.maxAddrLen(skt.fileno())
    self.addressBuffer = _iocp.AllocateReadBuffer(addrLen)
    # WSARecvFrom takes an int
    self.addressLengthBuffer = _iocp.AllocateReadBuffer(
        struct.calcsize('i'))
 
  def startListening(self):
    """
    Create and bind my socket, and begin listening on it.
 
    This is called on unserialization, and must be called after creating a
    server to begin listening on the specified port.
    """
    self._bindSocket()
    self._connectToProtocol()
 
 
  def createSocket(self):
    return self.reactor.createSocket(self.addressFamily, self.socketType)
 
 
  def _bindSocket(self):
    try:
      skt = self.createSocket()
      skt.bind((self.interface, self.port))
    except socket.error, le:
      raise error.CannotListenError, (self.interface, self.port, le)
 
    # Make sure that if we listened on port 0, we update that to
    # reflect what the OS actually assigned us.
    self._realPortNumber = skt.getsockname()[1]
 
    log.msg("%s starting on %s" % (
        self._getLogPrefix(self.protocol), self._realPortNumber))
 
    self.connected = True
    self.socket = skt
    self.getFileHandle = self.socket.fileno

    難道說twisted就完全不提供這樣的功能?最終在multicast中翻到這樣一段,也就是,多播的情況是支持地址復用的,動手測起來。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class MulticastPort(MulticastMixin, Port):
  """
  UDP Port that supports multicasting.
  """
 
  implements(interfaces.IMulticastTransport)
 
 
  def __init__(self, port, proto, interface='', maxPacketSize=8192,
         reactor=None, listenMultiple=False):
    Port.__init__(self, port, proto, interface, maxPacketSize, reactor)
    self.listenMultiple = listenMultiple
 
 
  def createSocket(self):
    skt = Port.createSocket(self)
    if self.listenMultiple:
      skt.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
      if hasattr(socket, "SO_REUSEPORT"):
        skt.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1)
    return skt

     將server端改成如下代碼,運行通過!

?
1
2
reactor.listenMulticast(port, DhtResponseHandler(queue), listenMultiple=True)
reactor.run()

     感觸良多,底層的知識比較重要,浮沙筑高臺果然危險。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: a v在线男人的天堂观看免费 | 日本不卡1卡2卡三卡网站二百 | 精品亚洲视频在线观看 | 92国产福利视频一区二区 | 香蕉人人超人人超碰超国产 | 国产东北3p真实在线456视频 | 亚洲天堂精品在线 | 国产第一自拍 | 91久久青青草原线免费 | 亚洲系列第一页 | 青草碰人人澡人人澡 | 亚洲日本在线观看网址 | 冰雪奇缘1完整版免费观看 变形金刚第一部 | 国内精品久久久久小说网 | 欧美一区a | 短篇艳妇系列 | 日本亚欧乱色视频在线观看 | 九九99九九精彩 | 日本一区二区三区在线 视频 | 国产剧情麻豆刘玥视频 | 婷综合 | 国产高清在线视频一区二区三区 | 色国产视频 | 性色AV乱码一区二区三区视频 | 国产福利一区二区在线精品 | 亚洲视频中文字幕 | 午夜办公室在线观看高清电影 | 久久er99热精品一区二区 | 999精品视频这里只有精品 | 精品久久久久免费极品大片 | 精品无码久久久久久久动漫 | 午夜国产福利视频一区 | 亚洲 欧美 中文 日韩欧美 | 国产va免费精品高清在线 | 欧美精品一区二区三区免费观看 | 日本乱中文字幕系列在线观看 | 特黄特黄一级片 | 午夜性爽视频男人的天堂在线 | 欧美作爱福利免费观看视频 | 日本 片 成人 在线 日b视频免费 | 五月天视频网 |