программирование сокетов (сервер и клиент на одном компьютере) что-то не так с соединением - PullRequest
1 голос
/ 14 апреля 2011

Я новичок в программировании сокетов, и у меня есть этот клиент, который пытается подключиться к серверу на том же компьютере.Но сервер зависает там после связывания или принятия, потому что связывание кажется правильным, но не выводится.Я знаю, что сервер работает, потому что другой клиент может нормально подключиться, и клиент, кажется, сделал это.Что заставляет сервер не видеть это входящее соединение?Я в полном здравом уме.

И я не привык к программированию на Mac, поэтому большое спасибо за терпение, если я допустил несколько глупых ошибок.

Мой код следующий:

server.cpp

    using namespace std;
    #include<iostream>
    #include <netinet/in.h>
    #include<stdio.h>
    #include<stdlib.h>
    #include<unistd.h>
    #include<sys/types.h>
    #include<sys/socket.h> 
    #include<arpa/inet.h>

    #define PORT 8888
    #define BACKLOG 20
    //#define DEST_IP "127.0.0.1"


    int process_conn_server(int s)
    {
        ssize_t size =0;
        char buffer[1024];

        for( ; ; )
        {
            size = read(s,buffer,1024);
            if(size == 0)
            {
                return 0;
            }
        }
        sprintf(buffer, "%d bytes altogether\n", (int)size);
        write(s, buffer,strlen(buffer)+1);



        return 0;
    }

    int main(int argc,char *argv[])
    {
        //cout<<"?";
        int ss, sc, r, err;
        struct sockaddr_in server_addr;
        struct sockaddr_in client_addr;
        int   opt=1; 
        pid_t pid;

        bzero(&server_addr, sizeof(server_addr));
        server_addr.sin_family = AF_INET;
        server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
        server_addr.sin_port = htons(PORT);

        ss = socket(AF_INET, SOCK_STREAM, 0);
        if(ss<0)
        {
            cout<<"[process infro]socket error"<<endl;
            return -1;
        }
        cout<<"[process infro]socket successful"<<endl;

        r = setsockopt(ss, SOL_SOCKET,SO_REUSEADDR, (void*)&opt,sizeof(opt));
        if (r == -1)
        { 
            perror("setsockopt(listen)"); 
            return 0;
        }
        cout<<"[process infro]sockopt successful"<<endl;

        cout<<"?";
        err = bind(ss, (struct sockaddr*) &server_addr, sizeof( server_addr));
        cout<<"err";
        if(err < 0)
        {
            cout<<"[process infro]bind error"<<endl;
            return -1;
        }
        cout<<"[process infro]bind successful";


        err=listen(ss, BACKLOG);
        if(err <0)
        {
            cout<<"[process infro]listen error"<<endl;
            return -1;
        }
        cout<<"[process infro]lisen successful";

        for( ; ; )
        {
            int addrlen = sizeof(struct sockaddr);

            sc = accept(ss, (struct sockaddr*)&client_addr, (socklen_t *)&addrlen);

            if(sc < 0)
            {
                continue;
            }

            pid =  fork();
            if (pid == 0)
            {
                close(ss);
                process_conn_server(sc);
            }
            else
            {
                close(sc);
            }
        }
        //opt=0; 

        //setsockopt(ss,SOL_SOCKET,SO_REUSEADDR,(void*)&opt,sizeof(len));
    }

client.cpp

    using namespace std;
    #include<iostream>
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <sys/socket.h>
    #include <unistd.h>
    #include <sys/types.h>
    #include <netinet/in.h>
    #include <time.h>
    #include <arpa/inet.h>
    #include <fstream.h>

    #define PORT 8888
    #define DEST_IP "127.0.0.1"

    void process_conn_client(int s)
    {
        ssize_t size = 0;
        char buffer[1024];

        //read from the file to be sent
        fstream outfile("programm.txt",ios::in|ios::out);       

        if (outfile.fail())
        {
            printf("[process infro]cannot open the file to be sent\n");
            return ;
        }   
        printf("[process infro]successfully open the file to be sent\n");

        while(!outfile.eof())
        {

            outfile.getline(buffer,1025,'\n');
            write(s,buffer,1024);       
            size = read(s, buffer, 1024);
            if(size = 0)
            {
                return ;
            }
            //write to the server
            write(s,buffer,size);

            //get response from the server
            size=read(s,buffer,1024);
            write(1,buffer,size);

        }
        outfile.close();    //关闭文件
    }

    int main(int argc,char *argv[])
    {
        int s;
        struct sockaddr_in server_addr;
        bzero(&server_addr, sizeof(server_addr));
        server_addr.sin_family = AF_INET;
        server_addr.sin_addr.s_addr = inet_addr(DEST_IP);
        server_addr.sin_port =  htons(PORT);

        s = socket(AF_INET, SOCK_STREAM, 0);
        if(s < 0)
        {
            cout<<"[process infro]socke error"<<endl;
            return -1;
        }
        cout<<"[process infro] socket built successfully\n";



        //inet_pton(AF_INET, argv[1], &server_addr.sin_addr);


        connect(s, (struct sockaddr*)&server_addr, sizeof(struct sockaddr));
        cout<<"[process infor] connected\n";
        process_conn_client(s);

        close(s);

        return 0;
    }

1 Ответ

2 голосов
/ 14 апреля 2011

Это может быть не связано .... но это не помещается в комментарии ...

На вашем сервере вы делаете это:

    int process_conn_server(int s)
    {
        ssize_t size =0;
        char buffer[1024];

        for( ; ; )
        {

// продолжаем читать, пока чтение не вернется 0

            size = read(s,buffer,1024);
            if(size == 0)
            {
                return 0;
            }
        }
        sprintf(buffer, "%d bytes altogether\n", (int)size);
        write(s, buffer,strlen(buffer)+1);



        return 0;
    }

В вашем клиенте вы делаете это:

    void process_conn_client(int s)
    {
        ssize_t size = 0;
        char buffer[1024];

        //read from the file to be sent
        fstream outfile("programm.txt",ios::in|ios::out);       

        if (outfile.fail())
        {
            printf("[process infro]cannot open the file to be sent\n");
            return ;
        }   
        printf("[process infro]successfully open the file to be sent\n");

        while(!outfile.eof())
        {

            outfile.getline(buffer,1025,'\n');

// запись на сервер?

            write(s,buffer,1024);       

// читать с сервера?

            size = read(s, buffer, 1024);
            if(size = 0)
            {
                return ;
            }
            //write to the server
            write(s,buffer,size);

            //get response from the server
            size=read(s,buffer,1024);
            write(1,buffer,size);

        }
        outfile.close();    
    }

Из-за имен переменных вам сложно следовать, но похоже, что ваш клиент работает в предположении, что ваш сервер отправит ответ на каждый полученный кусок данных, а это не так. Ваш сервер, по-видимому, не изменил принятый сокет на неблокирующий, поэтому он будет блокировать вызов чтения, пока не будут прочитаны некоторые данные (он никогда не получит 0) ...

Вы уверены, до этого момента происходит сбой? У вас есть пример вывода?

Асо, на ваш звонок для принятия вы передаете адрлен ...

 int addrlen = sizeof(struct sockaddr);  

Я думаю, это должно быть:

 int addrlen = sizeof(struct sockaddr_in);   /* sizeof(client_addr) */
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...