gpt4 book ai didi

c++ - Winsock 发送失败,错误 10093

转载 作者:行者123 更新时间:2023-11-30 03:00:51 26 4
gpt4 key购买 nike

我正在编写一个 bittorrent 客户端,需要通过 TCP 连接联系多个跟踪器。为此,我编写了如下所示的 winsock 包装器类:

class trackerSocket{
public:
~trackerSocket();

int trackerInitialize(string address);
int trackerSend(string getParams);
int trackerRecv();

be_node *responseDict;
bool working;

private:
string address;
string port;
string protocol;
string page;
SOCKET ConnectSocket;

int parseAnnounce(string announce);
int parseTrackerResponse(string response);
};

程序首先将一个新的 trackerSocket 类分配给一个变量。在此函数上调用 trackerInitialize 函数,如果成功,该类将被推送到一个 vector 上以存储所有工作跟踪器。这是 trackerInitialize 函数:

int trackerSocket::trackerInitialize(string announce){
WSADATA wsaData;
int iResult;

working = true;

iResult = parseAnnounce(announce);
if(iResult != 0){
working = false;
return iResult;
}

//Initialize Winsock
iResult = WSAStartup(MAKEWORD(2,2), &wsaData);
if(iResult != 0){
return 1;
}

struct addrinfo *result = NULL,
*ptr = NULL,
hints;

ZeroMemory(&hints, sizeof(hints));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
hints.ai_protocol = IPPROTO_TCP;

//Resolve the server address and port
iResult = getaddrinfo(address.c_str(), port.c_str(), &hints, &result);
if(iResult != 0){
WSACleanup();
return 1;
}

ConnectSocket = INVALID_SOCKET;

//Attempt to connect to the first address returned by
//the call to getaddrinfo
ptr = result;

do{
//Create a socket for connecting to the server
ConnectSocket = socket(ptr->ai_family, ptr->ai_socktype, ptr->ai_protocol);
if(ConnectSocket == INVALID_SOCKET){
ptr = ptr->ai_next;
continue;
}

//Connect to server
iResult = connect(ConnectSocket, ptr->ai_addr, (int)ptr->ai_addrlen);
if(iResult != SOCKET_ERROR){
break;
} else {
closesocket(ConnectSocket);
ConnectSocket = INVALID_SOCKET;
ptr = ptr->ai_next;
}
} while(ptr != NULL);

freeaddrinfo(result);
if(ConnectSocket == INVALID_SOCKET){
working = false;
WSACleanup();
return 1;
}

return 0;
}

然后程序执行代码以生成要发送给跟踪器的消息。对于 vector 中的每个跟踪器类,它的 trackerSend 函数都被消息调用,这里是 trackerSend 函数:

int trackerSocket::trackerSend(string getParams){
int iResult;

ostringstream os;
os << "GET " << page << getParams << " HTTP/1.1\r\n"
<< "Host: " << address << "\r\n"
<< "Accept: text/html\r\n"
<< "\r\n";

string sendBuf = os.str();

//Send tracker request
iResult = send(ConnectSocket, sendBuf.c_str(), strlen(sendBuf.c_str()), 0);
if(iResult == SOCKET_ERROR){
working = false;
closesocket(ConnectSocket);
WSACleanup();
return 1;
}

return 0;
}

每次程序运行时,发送函数都会为每个跟踪器返回 -1。如果我调用 WSAGetLastError() 函数,它会返回 10093。此错误的 msdn 定义是:

尚未执行成功的 WSAStartup。应用程序未调用 WSAStartup 或 WSAStartup 失败。应用程序可能正在访问当前事件任务不拥有的套接字(即,试图在任务之间共享套接字),或者 WSACleanup 被调用了太多次。

我看不到 WSACleanup 被调用了太多次,所以我只能假设套接字不属于当前事件任务(不知道那是什么意思),任何人都可以看到问题吗?

这是主程序的一些代码(我在上面已经描述过了):

//Store tracker URL's in vector
vector<trackerSocket> trackers;
trackerSocket *temptracker = new trackerSocket();
iResult = temptracker->trackerInitialize(announce);
if(iResult == 0){
trackers.push_back(*temptracker);
}
if(announcelist != NULL){
i = 0;
while(announcelist[i]){
if(strcmp(announcelist[i]->val.l[0]->val.s, announce.c_str()) != 0){
temptracker = new trackerSocket();
iResult = temptracker->trackerInitialize(announcelist[i]->val.l[0]->val.s);
if(iResult == 0){
trackers.push_back(*temptracker);
}
}
i++;
}
}

//Check that at least one of the tracker URL's was valid
if(trackers.size() == 0){
printf("None of the tracker URL's provided were valid.\n");
return 1;
}

//Generate some required values
string peerid = genPeerID();
string peerport = "12345";
int uploaded = 0;
int downloaded = 0;

//Work out how many bytes are left to download
int left = 0;
if(singlefile){
left = length;
} else {
for(i = 0; i < filesinfo.size(); i++){
left += filesinfo[i].length;
}
}

//Send GET Request to tracker
i = 0;
ostringstream os;
string getParams;
string response;
os << "info_hash=" << infohash << "&peer_id=" << peerid << "&port=" << peerport <<
"&uploaded=" << uploaded << "&downloaded=" << downloaded << "&event=started";

getParams = os.str();
do{
iResult = trackers[i].trackerSend(getParams);
if(iResult != 0){
printf("trackerSend %d failed: %d\n", i, iResult);
i++;
continue;
}
} while(i < trackers.size());

最佳答案

根据您的代码,您不应在发送时调用 WSACleanup,如果一个跟踪器无法发送,您将减少内部计数器,如果它达到 0,系统将需要一个您未调用的新 WSAStartup...

我建议您只调用一次 WSAStartup(例如在应用程序开始时)和一次 WSACleanup。 (例如申请结束)

关于c++ - Winsock 发送失败,错误 10093,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11649259/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com