使用 TCP 套接字的 C++ 多客户端聊天

标签 c++ sockets tcp

我正在尝试使用 TCP 套接字在 C++ 中进行多客户端聊天。
我已经从 this site 下载了套接字实现的源代码.

问题是当我尝试从客户端向服务器发送消息时,
我从服务器收到的“ecko”是无穷无尽的空格字符串。

我试图调试客户端代码,客户端正确读取输入。
在前几条消息中,服务器将他的消息发送回客户端,
但经过几条消息后,客户端会返回无尽的空间。
我尝试使用 memset 来取消(在所有数组中放置零),但更糟糕的是,服务器根本不接收消息。

非常感谢帮助(:

这是服务器端:

 #include "PracticalSocket.h"
#include <stdio.h>
#include <process.h>
#include <Windows.h>
using namespace std;
#pragma comment(lib,"ws2_32.lib")


TCPSocket* MyClients[20];
int ClientCount = 0;

 void connectCLient(void* pValue){  
    int nI,Flag;
    char st[1024];
    //memset(st,0,1024); // doing problems
    TCPSocket* pServerClient = (TCPSocket*)pValue;  
    MyClients[ClientCount] = pServerClient;
    ClientCount++;
    try{
        while (true)
        {
             Flag = pServerClient->recv(st,strlen(st));
             if(Flag>1){
                printf("%s\n",st);
                for(nI = 0; nI< ClientCount ; nI++){
                    MyClients[nI]->send(st,strlen(st)+1);
             }
        }


        }
    }
    catch(...){
        puts("one client lefttt");
    }
 }

int main(int argc, char* argv[])
{
    TCPServerSocket* pServer = new TCPServerSocket(8546);
    int nClientCounter = 0;
    printf("Start TCP Server ... on Port %d\n", 8546);

    try{
        while(true)
        {

            printf("Wait for new TCP Clients ... \n");
            TCPSocket* pClient = pServer->accept();     
            _beginthread(connectCLient,0,(void*)pClient);
            printf("Client %d Connected ... \n", ++nClientCounter);
        }
    }
    catch(...){
        puts("one client left");
    }

    return 0;
}

这是客户端:
#include "PracticalSocket.h"
#include <stdio.h>
#include <process.h>
#include <Windows.h>
using namespace std;
#pragma comment (lib, "ws2_32.lib")


void ReciveMessages(void * pValue ){
    char recvM[1024];
    TCPSocket* pClient = (TCPSocket*)pValue;
    while(true){
        pClient->recv(recvM,strlen(recvM));
        printf("%s\n",recvM);
    }
}


int main(int argc, char* argv[])
{
    try
    {
        TCPSocket * cClient = new TCPSocket();
        cClient->connect("127.0.0.1",8546);
        _beginthread(ReciveMessages,0,(void*)cClient);
        char st[1024];
        memset(st,0,1024);
        while(true)
        {

            printf("Press Text -->");
            fgets(st, sizeof st, stdin);
            cClient->send(st,strlen(st)+2); 
        }
    }
    catch(...)
    {
        printf("Socket Error..!");
        system("pause");//run cmd comment - stop the system
    }
    return 0;
}

最佳答案

代码中有几个错误:

MyClients[ClientCount] = pServerClient;
ClientCount++;

由于上述情况发生在不同的线程中,ClientCount++是非原子的并导致竞争条件。制作 ClientCount原子或在一个服务器线程中执行此操作。

在:
Flag = pServerClient->recv(st,strlen(st));
if(Flag>1) {
    printf("%s\n",st);
    for(nI = 0; nI< ClientCount ; nI++)
        MyClients[nI]->send(st,strlen(st)+1);
st不以 \0 结尾,因为它可以是部分读取,所以strlen(st)返回错误的结果。使固定:
ssize_t received = pServerClient->recv(st, sizeof st - 1);
if(received > 0) {
    st[received] = 0; // Zero-terminate.        
    printf("%s\n", st);
    for(nI = 0; nI< ClientCount ; nI++)
        MyClients[nI]->send(st, received);

类似问题:
pClient->recv(recvM,strlen(recvM));
printf("%s\n",recvM);

使固定:
ssize_t received = pClient->recv(recvM, sizeof recvM - 1);
if(received > 0) {
    recvM[received] = 0;
    printf("%s\n",recvM);
}

在:
 cClient->send(st,strlen(st)+2); 

发送零终止符没有意义:
 cClient->send(st, strlen(st)); 

TCP 是一种流协议(protocol),这意味着 sendrecv可以发送/接收部分数据并且没有消息边界。您可能喜欢delimit your messages .

关于使用 TCP 套接字的 C++ 多客户端聊天,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48444496/

相关文章:

c - 需要帮助修复此 C Socket ICMP 程序

c++ - 尝试创建 UDP 服务器

c# - 加密文件并将其发送通过

sockets - 在 nginx 中使用 TCP_QUICKACK

python scapy : how to translate port numbers to service names?

c++ - 缩短 If 语句并减少冗余

c++ - 将 2D 数组传递给函数

c++ - 将 NULL 传递给 std::wstring 时如何防止运行时崩溃?

linux - 如何获取监听套接字的BACKLOG

c++ - 如何重载析构函数?