2

2 つのプログラムを接続するための中間として、次のコードを作成しました。2 つの異なるシステムで実行中のサーバー プログラムとクライアント プログラムがあります。このコードは、これら 2 つのプログラムの中間として機能することが期待されます。

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

//Connect with program on server side
char * serv_con(char app_data[50])
{
int sock, bytes_recieved;  
char send_data[1024];
char *recv_data = malloc(1024);
struct hostent *host;
struct sockaddr_in server_addr;  
host = gethostbyname("10.47.3.249");
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{

    perror("Socket");
    exit(1);

}
server_addr.sin_family = AF_INET;     
server_addr.sin_port = htons(3128);   
server_addr.sin_addr = *((struct in_addr *)host->h_addr);
bzero(&(server_addr.sin_zero),8); 
if (connect(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) 
{

    perror("Connect");
    exit(1);

}
bytes_recieved=recv(sock,recv_data,1024,0);
recv_data[bytes_recieved] = '\0';
send(sock, app_data, 50, 0);
return recv_data;
//close(sock);

}


//Connect with client app
char * cli_con(char ser_data[50])
{
int sock, connected, bytes_recieved , true = 1;  
char send_data [1024];
char *recv_data = malloc(1024);       
struct sockaddr_in server_addr,client_addr;    
int sin_size;
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{

    perror("Socket");
    exit(1);

}
if (setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&true,sizeof(int)) == -1)
{

    perror("Setsockopt");
    exit(1);

}
server_addr.sin_family = AF_INET;         
server_addr.sin_port = htons(5000);     
server_addr.sin_addr.s_addr = INADDR_ANY; 
bzero(&(server_addr.sin_zero),8); 
if (bind(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
{

    perror("Unable to bind1");
    exit(1);

}
if (listen(sock, 5) == -1)
{

    perror("Listen");
    exit(1);

}
sin_size = sizeof(struct sockaddr_in);
connected = accept(sock, (struct sockaddr *)&client_addr,&sin_size1);
printf("\n I got a connection from (%s , %d)",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
bytes_recieved = recv(connected,recv_data,1024,0);
recv_data[bytes_recieved] = '\0';   
send(connected, ser_data,50, 0);
//close(sock);

}


int main()
{

char *ser_data, *app_data;
int pid = fork();
while(1)
{

    if(pid == 0)
        app_data = serv_con(ser_data);

    else
        ser_data = cli_con(app_data);

}

}

クライアント側のアプリが実行されるまでは正常に動作します。ただし、クライアント側のアプリが実行されるとすぐに、コードが終了してエラーが発生します。

Unable to bind: Address already in use
I got a connection from (192.168.0.3 , 45691)

このエラーを修正するには、コードにどのような変更を加える必要がありますか? 私はLinuxに取り組んでいます。前もって感謝します。

編集: からコメントを削除し、関数close(sock)に追加しました。クライアント側のコードは次のとおりです。close(connect)cli_con

int sock, bytes_recieved;  
char send_data[1024],recv_data[1024];
struct hostent *host;
struct sockaddr_in server_addr;  
host = gethostbyname("192.168.0.2");
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
    perror("Socket");
    exit(1);
}
server_addr.sin_family = AF_INET;     
server_addr.sin_port = htons(5555);   
server_addr.sin_addr = *((struct in_addr *)host->h_addr);
bzero(&(server_addr.sin_zero),8); 
if (connect(sock, (struct sockaddr *)&server_addr,
sizeof(struct sockaddr)) == -1) 
{
    perror("Connect");
    exit(1);
}
while(1)
{
    //necessary codes
    if (connect(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) 
    {
        close(sock);
        goto connct;
    }
}

しかし、今実行すると、最初のプログラムは終了せず、印刷さえしません

I got a connection from (192.168.0.3 , 45691)

しかし、メッセージを出力せずに実行を続けます。しかし一方で、クライアントはエラーを表示して終了します。

Connect: Connection reset by peer

私は今どうすればいい?

4

3 に答える 3

3

クライアントが切断されたら、新しいサーバー ソケットを作成し、それを同じポートにバインドします。サーバー側のソケットが閉じられていない場合、ポートはまだ使用されているため、bind失敗します。

通常、ソケット プログラムのサーバー側には、accept多くのクライアントからの接続を処理できるようにするためのループがあります。このようbindにしlistenて、一度だけ呼び出されます。

while (connected = accept(sock, (struct sockaddr *)&client_addr,&sin_size1)) {
    printf("\n I got a connection from (%s , %d)",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
    bytes_recieved = recv(connected,recv_data,1024,0);
    recv_data[bytes_recieved] = '\0';   
    send(connected, ser_data,50, 0);
    close(connected);
}
于 2012-05-17T05:28:10.377 に答える
2

1 つの指針は、リッスン ソケットと、accept 呼び出しから取得したソケットの両方を閉じる必要があるということです。

close(connected);
close(sock);

また、ソケットは使用後に OS によって解放されるまでに時間がかかるため、「アドレスは既に使用されています」というエラーで失敗する場合があります。コードの SO_REUSEADDR 部分が適切に実行されているかどうかを、コードで再度適切に確認できます。

また、メイン関数自体にリッスン ソケット作成コードを追加して、それを引数として cli_con 関数に渡すこともできます。一般的に従うメカニズムは、1 つのリッスン ソケットを作成し、それを使用してクライアントからの複数の接続を受け入れることです。

元のコードで、ser_data と app_data の適切なメモリ割り当て、初期化などが行われることを願っています。

于 2012-05-17T05:54:40.193 に答える
2

これは、ポート 5000 でリッスンしようとしているが、そのポートでリッスンしているプログラムが既に存在することを意味します (おそらく、ポートを適切に閉じていない以前のバージョンのプログラムです)。ポート番号を別の値に変更するか、閉じます。ポートでリッスンしているすべてのアプリケーション。

Linux を使用しているため、root として "netstat -nlp" を使用して、どのプログラムがどのポートを開いているかを確認できます。

于 2012-05-17T05:27:43.340 に答える