Использование потока std::для отправки и получения данных из сокета TCP

#c #sockets #client-server #tcpclient

Вопрос:

Я разрабатываю клиент сокета TCP. Я пытаюсь написать клиент, у которого есть отдельный поток для отправки и получения данных. но когда я использую поток, появляется ошибка во время выполнения.

Процесс происходит следующим образом:

main.cpp:

 #include <iostream>
#include <thread>
#include <string>
#include <chrono>
#include <WS2tcpip.h>

#pragma comment(lib, "ws2_32.lib")

#include "TCPConnection.h"

SOCKET sock;

void SendRecive(SOCKET *isock) {
    // Do-While loop to send and recive data
    char buf[4096];

    while (true) {
        // Prompt the user for some text
        cout << "> ";
        // Wait for response
        ZeroMemory(buf, 4096);
        auto byteRecived = recv(*isock, buf, 4096, 0);
        if (byteRecived <= 0) continue;
        // Echo response to console
        cout << "SERVER> " << string(buf, 0, byteRecived) << endl;
    }
}

using namespace std;

int main() {
    TCPConnection tcpconn("192.168.1.4", 7705, amp;sock);
    SendRecive(amp;sock);
    return 0;
}
 

TCPConnection.h:

 //
// Created by Hamed on 8/21/2021.
//

#ifndef TCPCLIENT_TCPCONNECTION_H
#define TCPCLIENT_TCPCONNECTION_H

#include <iostream>
#include <thread>
#include <string>
#include <WS2tcpip.h>

#pragma comment(lib, "ws2_32.lib")

using namespace std;

class TCPConnection {
public:
    SOCKET *isock;
    TCPConnection(string ServerIPAddress, int ServerPort, SOCKET *sock);
    ~TCPConnection();
};

#endif //TCPCLIENT_TCPCONNECTION_H
 

TCPConnection.cpp:

 #include "TCPConnection.h"

TCPConnection::TCPConnection(string ServerIPAddress, int ServerPort, SOCKET *sock) {
    // Initialize WinSock
    WSAData data{};
    WORD ver = MAKEWORD(2, 2);
    int wsResult = WSAStartup(ver, amp;data);
    if (wsResult != 0) {
        cerr << "Can't start winsock, Err #" << wsResult << endl;
        return;
    }

    // Create socket
    *sock = socket(AF_INET, SOCK_STREAM, 0);
    if (*sock == INVALID_SOCKET) {
        cerr << "Can't create socket, Err #" << WSAGetLastError() << endl;
        WSACleanup();
        return;
    }

    // Fill in a hint structure
    sockaddr_in hint{};
    hint.sin_family = AF_INET;
    hint.sin_port = htons(ServerPort);
    inet_pton(AF_INET, ServerIPAddress.c_str(), amp;hint.sin_addr);

    // connect to server
    int connResult = connect(*sock, (sockaddr *) amp;hint, sizeof(hint));
    if (connResult == SOCKET_ERROR) {
        cerr << "Can't connect to server, Err #" << WSAGetLastError() << endl;
        closesocket(*sock);
        WSACleanup();
        return;
    }
    isock = sock;
}



TCPConnection::~TCPConnection() {
    // Gracefully close down everything
    closesocket(*isock);
    WSACleanup();
}
 

этот код работает правильно, но когда я изменяю «main.cpp» чтобы использовать такую нить, как эта:

 int main() {
    TCPConnection tcpconn("192.168.1.4", 7705, amp;sock);    
    thread DoRunTCPRecive(SendRecive,amp;sock);
    return 0;
}
 

Я получаю ошибку отладки «Библиотека среды выполнения Microsoft Visual C » при запуске приложения.

введите описание изображения здесь

Обновление: На самом деле я хочу иметь функцию для отправки и другую функцию для получения данных . но когда я использую join, мой код остается в текущей функции, и после этого я не могу запустить другой код.

Ответ №1:

вы должны присоединиться к потоку после его создания

     thread DoRunTCPRecive(SendRecive,amp;sock);
    DoRunTCPRecive.join()
 

Комментарии:

1. Я хочу запустить еще одну функцию наряду с этой функцией. на самом деле я хочу иметь функцию для отправки и другую функцию для получения данных . но когда я использую join, мой код остается в этой строке.

2. вы имеете в виду, что хотите отправлять данные в основном потоке?верно?

3. Нет, я хочу иметь две нити. один из них отправляет данные, а второй поток получает данные.

4. затем вы можете сделать так, как этот поток DoRunTCPRecive(SendRecive,amp;sock); поток DoRunTCPSend(…); DoRunTCPRecive.join();DoRunTCPSend.join();

5. @user3123721 Всякий раз, когда у вас есть поток, вам нужно либо дождаться его завершения (с присоединением), либо не ждать его завершения (с отсоединением).