Python:Socket不想关闭,setsockopt()被忽略?(问题调试)

0 投票
1 回答
559 浏览
提问于 2025-04-16 22:36

编辑过的内容

最初的问题是关于重新连接时遇到的麻烦(关闭和关闭连接的混淆)。下面的代码是修复后的可用代码。

对于搜索的朋友,这个脚本是一个IRC机器人。功能列表:

  • 会不断尝试重新连接,直到成功连接上
  • 如果指定的昵称已经被占用,就在名字后面加上一个字符串(一直重复这个过程,直到成功)
  • 会监听PING消息,并用PONG回应
  • 可以接收命令并做出回应
  • 如果连接丢失,机器人会尝试重新连接(如果在5分钟内没有收到任何信息,比如没有PING消息,就会认为连接已经断开)

大致就是这些 :)

完整代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
import sys
import socket
import string
import os
import platform
import time

# Variables
HOST = "irc.server.net"
PORT = 6667
NICK = "Nickname"
IDENT = "Nickname"
REALNAME = os.getenv('USER')
CHAN = "##Channel"
readbuffer = ""

# The Connection itself
keep_connecting = True
while keep_connecting:
    irc = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    irc.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
    irc.settimeout(300)
    try:
        irc.connect((HOST, PORT))
        pass
    except socket.gaierror:
        print "No connection, attempting to connect again"
        time.sleep(5)
        continue
    print "Sending info..."
    irc.send("NICK %s\r\n" % NICK)
    irc.send("USER %s %s bla :%s\r\n" % (IDENT, HOST, REALNAME))
    irc.send("JOIN :%s\r\n" % CHAN)
    # Initial msg to send when bot connects
    irc.send("PRIVMSG %s :%s\r\n" % (CHAN, "TehBot: "+ NICK + " Realname: " + REALNAME + " ."))
    while True:
        try:
            data = irc.recv(4096)
            print data
            # If disconneted from IRC
            if len(data) == 0:
                print "Length of data == 0 ?..."
                break
            # If Nick is in use
            if data.find (NICK + " :Nickname is already in use") != -1:
                NICK = NICK + str(time.time())[5:-3]
                break
            # Ping Pong so we don't get disconnected
            if data[0:4] == "PING":
                irc.send ("PONG " + data.split() [ 1 ] + "\r\n")
        except socket.timeout:
            print "Socket timeout!"
            irc.close()
            break

1 个回答

1

这很可能是因为你关闭了Wi-Fi,导致系统中没有这个接口,所以你会看到类似于无法分配请求的地址这样的错误。当你尝试连接一个不存在的本地地址时,就会出现这种错误。

另外,调用close后,你将无法在同一个插口上重新连接,因为这会释放与这个插口相关的所有资源。

撰写回答