gpt4 book ai didi

c++ - 在套接字之间发送消息

转载 作者:塔克拉玛干 更新时间:2023-11-03 07:58:24 24 4
gpt4 key购买 nike

我有 1 台使用 C++ 和 Unix 中的 c 套接字构建的服务器。客户端使用的是QT和自带的socket api。

服务器向客户端发送345字节的数据。

从服务器发送消息:

void Moderator::testSynch(){
int type = (int) SYNCRHONIZE_M;
//Update all connected clients with info about other clients
for(int i = 0; i<nrOfClients_; i++){
const TcpSocket &clientSocket = clients_[i].getSocket();
int clientID = clients_[i].getID();

int tempType = htonl(type);
int tempClientID = htonl(clientID);
int tempNrOfClients = htonl(funNrOfClients);

clientSocket.writeData((const char*) &tempType, sizeof(tempType));
clientSocket.writeData((const char*) &tempClientID, sizeof(tempClientID));
clientSocket.writeData((const char*) &tempNrOfClients, sizeof(tempNrOfClients));

for(int j = 0; j<nrOfClients; j++){ //Send info about connectecd clients

int tempLength = (int) clients_[j].getName().length();
int tempID = clients_[j].getID();
string tempName = clients_[j].getName();

tempID = htonl(tempID);
tempLength = htonl(tempLength);
clientSocket.writeData((const char*) &tempID, sizeof(tempID));
clientSocket.writeData((const char*) &tempLength, sizeof(tempLength));
clientSocket.writeData(tempName.c_str(), (int)tempName.length());

}
}
}

bool TcpSocket::writeData(const char* buffer, int length)const{
size_t bytesLeft = length;
ssize_t bytesWritten = 0;

while((bytesWritten = write(socketFD_, buffer, bytesLeft)) > 0){
bytesLeft -= bytesWritten;
buffer += bytesWritten;
}
return bytesLeft == 0;
}

在客户端读取消息:

 void ChatClient::readMessage(Message &message){

if(socket_->readData((char*) &type, sizeof(type))){
if(type == SYNCRHONIZE_M){
int nrOfUsers = 0;

socket_->readData((char*) &ID_, sizeof(ID_)); //Set the client ID that server gave us
socket_->readData((char*) &nrOfUsers, sizeof(nrOfUsers));

ID_ = ntohl(ID_);
nrOfUsers = ntohl(nrOfUsers);
qDebug("%s=%d", "nrOfUsers", nrOfUsers);
message.setMessageType(SYNCRHONIZE_M);
messageOK = true;
for(int i = 0; i<nrOfUsers; i++){ //Update client with all connected users to server
int userID = 0;
int nameLength = 0;

socket_->readData((char*) &userID, sizeof(userID));
socket_->readData((char*) &nameLength, sizeof(nameLength));

userID = ntohl(userID);
nameLength = ntohl(nameLength);

if(nameLength > 0){
qDebug("%s=%d", "nameLength", nameLength);
buffer = new char[nameLength];
socket_->readData(buffer, nameLength);

message.addUser(ConnectedUser(buffer, nameLength, userID));
delete [] buffer;
}
}
}
}
}

bool TcpSocket::readData(char* buffer, int length){
int bytesLeft = length;
int bytesRead = 0;

while((bytesRead = qSocket_->read(buffer, bytesLeft)) > 0){
bytesLeft -= bytesRead;
buffer += bytesRead;

}
return bytesLeft == 0;
}

我遇到的问题是有时来自服务器的整个消息无法立即使用。

例如,前 45 个字节在客户端可用。然后客户端尝试读取整个消息(345 字节),这会导致奇怪的行为。客户端完成读取后,下一个 300 字节立即可用。

在套接字之间发送消息的最佳方式是什么?另外,如何确定是否已收到整条消息?

最佳答案

您有一些只存在于脑海中的“消息”概念。您的代码中没有任何内容反射(reflect)了这一点。如果您的应用程序协议(protocol)涉及要发送的“消息”,那么您需要编写代码来发送消息,并根据您的协议(protocol)编写代码来接收消息消息的定义。 TCP 仅提供字节流,不会为应用程序将它们粘合在一起形成大于一个字节的任何内容。

关于c++ - 在套接字之间发送消息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14073792/

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