Многопоточная розетка в C - PullRequest
0 голосов
/ 08 октября 2019

Я пытаюсь отредактировать код, который я написал ранее, от создания сокета клиента и сервера, позволяя клиенту отправить сообщение на сервер, и сервер отвечает "Я получил это ..." + Сообщение. Это хорошо работает, но следующие шаги - это то, с чем у меня возникают проблемы.

Я хочу иметь возможность непрерывно отправлять сообщения, пока клиент не скажет «Выход», и поддерживать несколько клиентов.

У меня есть код, который работает для связи один на один клиент / сервер:

Сервер

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

void error(const char *msg) {
    perror(msg);
    exit(1);
}

int main(int argc, char *argv[]) {
    int sockfd, newsockfd, portno;
    socklen_t clilen;
    char buffer[256];
    struct sockaddr_in serv_addr, cli_addr;
    int n; // return value for the read() and write() calls

    // gcc server.c -o server
    // ./server 8080
    // Pass in Port # or pass error
    if (argc < 2) {
        fprintf(stderr,"ERROR, no port provided\n");
        exit(1);
    }

    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    // Error if Socket Call fails
    if (sockfd < 0) 
    error("ERROR opening socket");
    bzero((char *) &serv_addr, sizeof(serv_addr)); // Set all values in a buffer to zero, args: pointer to buffer, size of buffer, sets serv_addr ot zero
    portno = atoi(argv[1]); // Port number listens for connections
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = INADDR_ANY;
    serv_addr.sin_port = htons(portno);

    // Binds socket to an address
    if (bind(sockfd, (struct sockaddr *) &serv_addr,
            sizeof(serv_addr)) < 0) 
            error("ERROR on binding");

    // Listens to the socket for connections
    listen(sockfd,5);

    clilen = sizeof(cli_addr);
    newsockfd = accept(sockfd, 
                (struct sockaddr *) &cli_addr, 
                &clilen);
    if (newsockfd < 0) 
    error("ERROR on accept");

    bzero(buffer,256);
    n = read(newsockfd,buffer,255);
    if (n < 0) error("ERROR reading from socket");
    printf("Here is the message: %s\n",buffer);


    n = write(newsockfd,"I got your message",18);
    if (n < 0) error("ERROR writing to socket");
    close(newsockfd);
    close(sockfd);
    return 0; 
}

Клиент:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>  



void error(const char *msg) {
    perror(msg);
    exit(0);
}

int main(int argc, char *argv[]) {
    int sockfd, portno, n;
    struct sockaddr_in serv_addr;
    struct hostent *server;
    char buffer[256];
    // gcc client.c -o client
    // ./client localhost 8080
    if (argc < 3) {
       fprintf(stderr,"usage %s hostname port\n", argv[0]);
       exit(0);
    }
    portno = atoi(argv[2]);
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) 
        error("ERROR opening socket");

    server = gethostbyname(argv[1]);
    if (server == NULL) {
    fprintf(stderr,"ERROR, no such host\n");
        exit(0);
    }
    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    bcopy((char *)server->h_addr, 
         (char *)&serv_addr.sin_addr.s_addr,
         server->h_length);
    serv_addr.sin_port = htons(portno);
    // Error Connecting to the server
    if (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0) 
        error("ERROR connecting");

    printf("Please enter the message: ");
    bzero(buffer,256);
    fgets(buffer,255,stdin);
    n = write(sockfd,buffer,strlen(buffer));
    if (n < 0) 
         error("ERROR writing to socket");
    bzero(buffer,256);
    n = read(sockfd,buffer,255);
    if (n < 0) 
         error("ERROR reading from socket");
    printf("%s\n",buffer);

    close(sockfd);
    return 0;
}

Мои мысли были для файла сервера естьцикл while:

while(buffer != "EXIT"){
    n = read(newsockfd,buffer,255);
    if (n < 0) error("ERROR reading from socket");
    printf("Here is the message: %s\n",buffer);


    n = write(newsockfd,"I got your message",18);
    if (n < 0) error("ERROR writing to socket");
    close(newsockfd);
    close(sockfd);
}

, но он выполняет только одно взаимодействие между одним клиентом и сервером.

1 Ответ

0 голосов
/ 08 октября 2019

Это не многопоточное приложение. Основная проблема, которую я вижу здесь, заключается в том, что ваш сервер не готов принять несколько клиентов одновременно. Цикл while будет обрабатывать только чтение и запись по одному.

Базовая логика, которую необходимо реализовать:

1.) Примите новый запрос на подключение, используя функцию «accept».

2.) Создайте новый поток (используя pthread_create) для чтения и записи в один и тот же запрос соединения. и одновременно подготовьтесь к новому запросу на соединение от функции «accept».

3.) Выполните шаги 1 и 2, чтобы сохранить поток отправки и получения до тех пор, пока не получите «EXIT».

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...