Python Interpreter блокирует многопоточные DNS-запросы?
Я немного поиграл с python и потоками и реализовал даже в многопоточном script, DNS-запросы блокируют. Рассмотрим следующий script:
from threading import Thread
import socket
class Connection(Thread):
def __init__(self, name, url):
Thread.__init__(self)
self._url = url
self._name = name
def run(self):
print "Connecting...", self._name
try:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setblocking(0)
s.connect((self._url, 80))
except socket.gaierror:
pass #not interested in it
print "finished", self._name
if __name__ == '__main__':
conns = []
# all invalid addresses to see how they fail / check times
conns.append(Connection("conn1", "www.2eg11erdhrtj.com"))
conns.append(Connection("conn2", "www.e2ger2dh2rtj.com"))
conns.append(Connection("conn3", "www.eg2de3rh1rtj.com"))
conns.append(Connection("conn4", "www.ege2rh4rd1tj.com"))
conns.append(Connection("conn5", "www.ege52drhrtj1.com"))
for conn in conns:
conn.start()
Я точно не знаю, сколько времени занимает время ожидания, но при выполнении этого происходит следующее:
- Все темы начинаются, и я получаю распечатки
- Каждый xx секунд, один поток отображается завершенным, а не все одновременно
- Потоки завершаются последовательно, не все сразу (timeout = одинаково для всех!)
Так что я только предполагаю, что это связано с GIL? Очевидно, что потоки не выполняют свою задачу одновременно, одновременно выполняется только одно соединение.
Кто-нибудь знает об этом?
(asyncore не помогает, и я бы предпочел не использовать скрученный)
Невозможно ли получить эту простую мелочь с помощью python?
Привет, Том
Изменить
Я на MacOSX, я просто позволю моему другу запустить это на linux, и он действительно получил результаты, которые я хотел получить. Его socket.connects() немедленно возвращается, даже в среде без резьбы. И даже когда он устанавливает блокировки на блокировку и тайм-аут до 10 секунд, все его потоки заканчиваются одновременно.
Кто-нибудь может это объяснить?
Ответы
Ответ 1
В некоторых системах getaddrinfo не является потокобезопасным. Python считает, что некоторые такие системы - FreeBSD, OpenBSD, NetBSD, OSX и VMS. В этих системах Python поддерживает блокировку специально для netdb (т.е. Getaddrinfo и друзей).
Итак, если вы не можете переключать операционные системы, вам придется использовать другую (потокобезопасную) библиотеку распознавателей, например, скрученные.
Ответ 2
если он подходит, вы можете использовать модуль multiprocessing
для включения процесса parallelism
import multiprocessing, socket
NUM_PROCESSES = 5
def get_url(url):
try:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setblocking(0)
s.connect((url, 80))
except socket.gaierror:
pass #not interested in it
return 'finished ' + url
def main(url_list):
pool = multiprocessing.Pool( NUM_PROCESSES )
for output in pool.imap_unordered(get_url, url_list):
print output
if __name__=="__main__":
main("""
www.2eg11erdhrtj.com
www.e2ger2dh2rtj.com
www.eg2de3rh1rtj.com
www.ege2rh4rd1tj.com
www.ege52drhrtj1.com
""".split())
Ответ 3
Отправлять DNS-запросы асинхронно, используя Twisted Names:
import sys
from twisted.internet import reactor
from twisted.internet import defer
from twisted.names import client
from twisted.python import log
def process_names(names):
log.startLogging(sys.stderr, setStdout=False)
def print_results(results):
for name, (success, result) in zip(names, results):
if success:
print "%s -> %s" % (name, result)
else:
print >>sys.stderr, "error: %s failed. Reason: %s" % (
name, result)
d = defer.DeferredList(map(client.getHostByName, names), consumeErrors=True)
d.addCallback(print_results)
d.addErrback(defer.logError)
d.addBoth(lambda _: reactor.stop())
reactor.callWhenRunning(process_names, """
google.com
www.2eg11erdhrtj.com
www.e2ger2dh2rtj.com
www.eg2de3rh1rtj.com
www.ege2rh4rd1tj.com
www.ege52drhrtj1.com
""".split())
reactor.run()