C中的UDP套接字:client1到服务器到client2

时间:2013-02-24 17:58:34

标签: c sockets udp

我对网络编程非常陌生,所以对我很陌生。到目前为止,我有一个UDP客户端/服务器,它以较低或大写的形式向服务器发送消息。服务器接收消息并通过切换的情况将其中继。我无法弄清楚的是我如何将服务器中继回第一个客户端,而不是将其发送到client2。如果我的client2发送一条消息,服务器接收并将其发送回client2,与client1相同。我想将client1所说的内容发送到服务器,然后服务器将其发送到client2。我已经尝试了我能想到的一切,但无法弄清楚。

服务器:

/*
Simple udp server

*/
#include<stdio.h>   //printf
#include<string.h> //memset
#include<stdlib.h> //exit(0);
#include<arpa/inet.h>
#include<sys/socket.h>
#include<ctype.h>

#define BUFLEN 512  //Max length of buffer
#define PORT 8888   //The port on which to listen for incoming data

void die(char *s)
{
    perror(s);
exit(1);
}

int main(void)
{
struct sockaddr_in si_me, si_other, si_other2;

int s, i, slen = sizeof(si_other) , recv_len;
char buf[BUFLEN];

//create a UDP socket
if ((s=socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) == -1)
{
    die("socket");
}

// zero out the structure
memset((char *) &si_me, 0, sizeof(si_me));

si_me.sin_family = AF_INET;
si_me.sin_port = htons(PORT);
si_me.sin_addr.s_addr = htonl(INADDR_ANY);

//bind socket to port
if( bind(s , (struct sockaddr*)&si_me, sizeof(si_me) ) == -1)
{
    die("bind");
}

//keep listening for data
while(1)
{
    printf("Waiting for data...");
    fflush(stdout);

    //try to receive some data, this is a blocking call
    if ((recv_len = recvfrom(s, buf, BUFLEN, 0, (struct sockaddr *) &si_other, &slen)) == -1)   // read datagram from server socket
    {
        die("recvfrom()");
    }

    //print details of the client/peer and the data received
    printf("Received packet from %s:%d\n", inet_ntoa(si_other.sin_addr), ntohs(si_other.sin_port));         printf("Data: %s\n" , buf);

    //now reply to server socket/the client with the same data
    if (sendto(s, buf, recv_len, 0, (struct sockaddr*) &si_other, slen) == -1)
    {
        die("sendto()");
    }



}

close(s);
return 0;
}

客户:

/*
Simple udp client

*/
#include<stdio.h>   //printf
#include<string.h> //memset
#include<stdlib.h> //exit(0);
#include<arpa/inet.h>
#include<sys/socket.h>
#include<ctype.h>

#define SERVER "192.x.x.x"
#define BUFLEN 512  //Max length of buffer
#define PORT 8888   //The port on which to send data

void die(char *s)
{
perror(s);
exit(1);
}

int main(void)
{
struct sockaddr_in si_other;
int s, s2, i, slen=sizeof(si_other);
char buf[BUFLEN];
char message[BUFLEN];

if ( (s=socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) == -1)        // create a client socket
{
    die("socket");
}

memset((char *) &si_other, 0, sizeof(si_other));
si_other.sin_family = AF_INET;
si_other.sin_port = htons(PORT);

if (inet_aton(SERVER , &si_other.sin_addr) == 0)            // Create datagram with server IP and port.
{
    fprintf(stderr, "inet_aton() failed\n");
    exit(1);
}

while(1)
{
    printf("Enter message : ");
    gets(message);


    int a;
    char message2[BUFLEN];
    for(a=0;a<=BUFLEN-1;a++)
      {
        if(message[a] >= 97 && message[a] <= 122)
           message2[a] = toupper(message[a]);
        else
           message2[a] = tolower(message[a]);

      }


    if (sendto(s, message2, strlen(message2) , 0 , (struct sockaddr *) &si_other, slen)==-1)
    {
        die("sendto()");
    }


    //receive a reply and print it
    //clear the buffer by filling null, it might have previously received data
    memset(buf,'\0', BUFLEN);
    //try to receive some data, this is a blocking call
    if (recvfrom(s, buf, BUFLEN, 0, (struct sockaddr *) &si_other, &slen) == -1)        // read datagram from client socket
    {
        die("recvfrom()");
    }

    puts(buf);
}

close(s);
return 0;
}

2 个答案:

答案 0 :(得分:1)

似乎不是UDP服务器,你想拥有一个像client-gt; server-&gt; client2那样的数据包转发器,所以你需要告诉你的服务器client2的ipaddress /端口,这样才能配置client2服务器中的ip / port使用命令行参数或读取任何输入文件,然后在服务器代码中的sendto语句之前,使用命令行或输入文件中的client *信息填充si_other结构。

答案 1 :(得分:1)

在您的read()recvfrom()消息之后,您处理数据然后sendto()两次:一个到recvfrom()(原始发件人)返回的地址和其他地址到另一个客户的地址(必须由您的服务器以某种方式提供或检测)。

此处的一些事情(正确的错误检查未执行):

char data[100];
struct sockaddr_in address;
socklen_t length = sizeof address;

/* Receive data from any client. */
ssize_t result = recvfrom(server, data, sizeof data, 0, &address, &length);

/* Process the data (change cases). */
process_data(data, result);

/* Send back to the first client. */
sendto(server, data, result, 0, &address, sizeof address);

/* Check who's the sender and relay to the other. */
if (address.sin_addr.s_addr == CLIENT1_ADDRESS)
    address.sin_addr.s_addr = CLIENT2_ADDRESS;
else
    address.sin_addr.s_addr = CLIENT1_ADDRESS;

/* Send to the other client. */
sendto(server, data, result, 0, &address, sizeof address);

在此示例中,地址是静态定义的。