답변:
일반적인 방법은 select () 를 사용 하여 데이터를 사용할 수있을 때까지 또는 시간 초과가 발생할 때까지 기다리는 것입니다. recv()
데이터를 실제로 사용할 수있을 때만 호출 합니다. 안전을 위해 소켓을 비 차단 모드로 설정하여 recv()
무기한 차단되지 않도록 보장 합니다. select()
한 번에 둘 이상의 소켓을 대기하는 데 사용할 수도 있습니다.
import select
mysocket.setblocking(0)
ready = select.select([mysocket], [], [], timeout_in_seconds)
if ready[0]:
data = mysocket.recv(4096)
열린 파일 설명 자가 많으면 poll () 이 select()
.
또 다른 옵션은을 사용하여 소켓의 모든 작업에 대한 시간 제한을 설정하는 socket.settimeout()
것이지만 다른 답변에서 해당 솔루션을 명시 적으로 거부 한 것을 확인했습니다.
select
은 좋지만 "할 수 없습니다"라고 말하는 부분은 오해의 소지가 socket.settimeout()
있습니다.
select
은 Windows 시스템에서 실행중인 경우 일부 데이터가 도착 select
하자마자 반환하는 습관이 있지만 반드시 전부 는 아니므 로 WinSock 라이브러리에 의존 합니다. 따라서 모든 데이터가 수신 될 때까지 계속 호출하려면 루프를 통합해야합니다 . 모든 데이터를 얻었는지 아는 방법은 (불행히도) 당신에게 달려 있습니다. 종결 자 문자열, 특정 바이트 수를 찾거나 정의 된 시간 초과를 기다리는 것을 의미 할 수 있습니다. select.select()
ready[0]
서버의 응답에는 몸이 존재하지 않는 경우에만 거짓?
select
이 솔루션이 하나의 라이너 (유지하기 쉽고 잘못 구현할 위험이 적음)이고 후드 아래에서 선택을 사용할 때 사용하는 솔루션 이 선호되는 이유를 모르겠습니다 (구현은 @DanielStuzbach 답변과 동일합니다).
으로 모두 언급 select.select()
하고 socket.settimeout()
작동합니다.
필요에 따라 settimeout
두 번 전화해야 할 수도 있습니다 . 예 :
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(("",0))
sock.listen(1)
# accept can throw socket.timeout
sock.settimeout(5.0)
conn, addr = sock.accept()
# recv can throw socket.timeout
conn.settimeout(5.0)
conn.recv(1024)
.settimeout()
한 번 이상 당신이 부를 수있는 setdefaulttimeout()
첫번째 장소에있는 방법을.
찾고있는 시간 초과는 서버 측을 구현하는 경우 기본 소켓이 아닌 연결 소켓의 시간 초과입니다. 즉, socket.accept()
method 의 출력 인 연결 소켓 개체에 대한 또 다른 제한 시간이 있습니다. 따라서:
sock.listen(1)
connection, client_address = sock.accept()
connection.settimeout(5) # This is the one that affects recv() method.
connection.gettimeout() # This should result 5
sock.gettimeout() # This outputs None when not set previously, if I remember correctly.
클라이언트 측을 구현하면 간단합니다.
sock.connect(server_address)
sock.settimeout(3)
이전 회신에서 언급했듯이 다음과 같이 사용할 수 있습니다. .settimeout()
예 :
import socket
s = socket.socket()
s.settimeout(1) # Sets the socket to timeout after 1 second of no activity
host, port = "somehost", 4444
s.connect((host, port))
s.send("Hello World!\r\n")
try:
rec = s.recv(100) # try to receive 100 bytes
except socket.timeout: # fail after 1 second of no activity
print("Didn't receive data! [Timeout]")
finally:
s.close()
이게 도움이 되길 바란다!!
socket.settimeout()
초 수를 나타내는 정수 인수를 허용하는 것을 사용할 수 있습니다 . 예를 들어 socket.settimeout(1)
시간 제한을 1 초로 설정합니다.
이것은 기본 C를 사용합니다.
timeval = struct.pack('ll', 2, 100)
s.setsockopt(socket.SOL_SOCKET, socket.SO_RCVTIMEO, timeval)
SO_RCVTIMEO
와 SO_SNDTIMEO
.
2
그리고 왜 100
? 시간 초과 값은 무엇입니까? 어떤 단위로?
timeval = struct.pack('ll', sec, usec)
s.setsockopt(socket.SOL_SOCKET, socket.SO_RCVTIMEO, timeval)
usec = 10000은 10ms를 의미합니다.
#! /usr/bin/python3.6
# -*- coding: utf-8 -*-
import socket
import time
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
s.settimeout(5)
PORT = 10801
s.bind(('', PORT))
print('Listening for broadcast at ', s.getsockname())
BUFFER_SIZE = 4096
while True:
try:
data, address = s.recvfrom(BUFFER_SIZE)
except socket.timeout:
print("Didn't receive data! [Timeout 5s]")
continue
https://boltons.readthedocs.io/en/latest/socketutils.html로 외칩니다.
버퍼링 된 소켓을 제공하며 다음과 같은 매우 유용한 기능을 많이 제공합니다.
.recv_until() #recv until occurrence of bytes
.recv_closed() #recv until close
.peek() #peek at buffer but don't pop values
.settimeout() #configure timeout (including recv timeout)