我在下面粘贴我的客户端和服务器代码。我的程序运行正常,除了我尝试在src和dest字段中发送ipaddress,并且由于某种原因,即使我将其发送为131.199.166.232,也正在将其打印为232.166.199.131。但是其余的数据包值以适当的方式打印。我使用过memcpy(),所以感觉到它是个memcpy东西,我在某个地方做错了,但是在Beej的指南中,因为在不同的计算机体系结构中有一个@字节排序节.....我没有使用过htonl()以及所有这些,也许就是因为这个原因。请向我指出我要去的地方。还请告诉我我发送数据的方式,应该如何在我的代码中使用htonl()函数……。谢谢。

客户:

       #include <stdio.h>
     #include <stdlib.h>
     #include <string.h>
     #include <math.h>
     #include <sys/types.h>
     #include <sys/socket.h>
     #include <netinet/in.h>
     #include <netdb.h>
     #define MAXPROFILES  2

    int main(int argc, char *argv[])

  {
    int sockfd, portno, n;
    struct sockaddr_in serv_addr;
    struct hostent *server;
    unsigned char buf[1024];
    unsigned int srcAddress = 2193598184;
    unsigned int destAddress = 2193598182;

   struct profile_t
 {
   unsigned char length;
   unsigned char type;
   unsigned char *data;
 };

   typedef struct profile_datagram_t
 {
unsigned char src[4];
unsigned char dst[4];
unsigned char ver;
unsigned char n;
struct profile_t profiles[MAXPROFILES];
 } header;


    header outObj;

    int j =0;
    int i =0;
   // for loop for doing the malloc so that we can allocate memory to all profiles
    for(i=0;i<MAXPROFILES;i++){
outObj.profiles[i].data = malloc(5);
  }


    for(i=3;i>=0;i--){
outObj.src[i] = (srcAddress >> (i*8)) & 0xFF;
outObj.dst[i] = (destAddress >> (i*8)) & 0xFF;
printf("%d",outObj.src[i]);
 }
    outObj.ver = 1;
    outObj.n = 2;

    memcpy(buf,&outObj.src,4);
    memcpy(buf+4,&outObj.dst,4);
    memcpy(buf+8,&outObj.ver,1);
    memcpy(buf+9,&outObj.n,1);


    outObj.profiles[0].length = 5;
    outObj.profiles[0].type = 1;
    outObj.profiles[1].length = 5;
    outObj.profiles[1].type = 2;

    for(i=0;i<MAXPROFILES;i++){
for(j=0;j<5;j++){
    outObj.profiles[i].data[j] = j+1;
}

}
    int k = 10;

        // for loop to do memcopy of length,type and data.
    for(i=0;i<MAXPROFILES;i++){
memcpy(buf+k,&outObj.profiles[0].length,1);
memcpy(buf+k+1,&outObj.profiles[0].type,1);
memcpy(buf+k+2,outObj.profiles[0].data,5);
k +=7;

}
   if (argc < 3) {
   fprintf(stderr,"usage: %s hostname port\n", argv[0]);
   exit(0);
}
   portno = atoi(argv[2]); //Convert ASCII to integer
   sockfd = socket(AF_INET, SOCK_STREAM, 0); // socket file descriptor


   if (sockfd < 0)
   error("ERROR DETECTED !!! Problem in opening socket\n");

   server = gethostbyname(argv[1]);
   if (server == NULL) {
   fprintf(stderr,"ERROR DETECTED !!!, no such server found \n");
    exit(0);
}

  bzero((char *) &serv_addr, sizeof(serv_addr)); //clear the memory for server address

  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);

  printf("Client 1 trying to connect with server host %s on port %d\n", argv[1], portno);


  if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
  error("ERROR in connection");

      printf("SUCCESS !!! Connection established \n");


   if (write(sockfd, buf, k) < 0)
   {
  error("Write error has occured ");
   }


  return 0;

服务器:
          #include <stdio.h>
           #include <stdlib.h>
           #include <string.h>
           #include <sys/types.h>
           #include <sys/socket.h>
           #include <netinet/in.h>
           #define MAXPROFILES  2

           int main(int argc, char *argv[])
        {
           int sockfd, newsockfd, portno, clilen;
           struct sockaddr_in serv_addr, cli_addr;
           unsigned char buf[1024];
           int my_data2[10] = {1,3,9,10};
           int my_data[10] = {1,2,3,4,5};
       int myDataBinary[500] = {0};
       int myDataBinary2[500] = {0};
       int recData[500] = {0};
       int index1=0;


       struct profile_t
        {
        unsigned char length;
        unsigned char type;
        unsigned char *data;
        };

            typedef struct profile_datagram_t
       {
        unsigned char src[4];
        unsigned char dst[4];
        unsigned char ver;
        unsigned char n;
       struct profile_t profiles[MAXPROFILES];
        } header;


            header outObj;

        int j =0;
        int i =0;



            if (argc < 2) {
            fprintf(stderr,"usage: %s port_number1",argv[0]);
            exit(1);
         }
            sockfd = socket(AF_INET, SOCK_STREAM, 0);
            if (sockfd < 0)
            error("ERROR DETECTED !!! Problem in opening socket");

            bzero((char *) &serv_addr, sizeof(serv_addr));
            portno = atoi(argv[1]);

            serv_addr.sin_family = AF_INET;
            serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
            serv_addr.sin_port = htons(portno);

            if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0)
            error("ERROR DETECTED !!! There was a problem in binding");

            listen(sockfd, 10);
            clilen = sizeof(cli_addr);



             printf("Server listening on port number %d...\n", serv_addr.sin_port);

             newsockfd = accept(sockfd,(struct sockaddr *) &cli_addr, &clilen);

             if (newsockfd < 0)
         error("ERROR DETECTED !!! the connection request was not accepted");

         int rc = read(newsockfd,buf,100);
         if(rc < 0){
     printf("error");
      }
         else {
        printf("success %d",rc);
      }


      memcpy(&outObj.src,buf+0,4);
      memcpy(&outObj.dst,buf+4,4);
      memcpy(&outObj.ver,buf+8,1);
      memcpy(&outObj.n,buf+9,1);

          printf("\nsrc ip = ");
          for(int i=0;i<4;i++){
       printf("%d ",outObj.src[i]);
       }

          printf("\ndest ip = ");
          for(int i=0;i<4;i++){
         printf("%d ",outObj.src[i]);
        }

         printf("\nversion = %d",outObj.ver);
         printf("\nnumber = %d",outObj.n);

         int k = 10;

         for(i=0;i<outObj.n;i++){
     memcpy(&outObj.profiles[i].length,buf+k,1);
     memcpy(&outObj.profiles[i].type,buf+k+1,1);
     outObj.profiles[i].data = malloc(outObj.profiles[i].length);
     memcpy(outObj.profiles[i].data,buf+k+2,5);
        k +=7;
       }

         for(int i=0;i<outObj.n;i++){
    printf("\nMessage %d :",i+1);
    printf("\nLength : %d",outObj.profiles[i].length);
    printf("\nType : %d",outObj.profiles[i].type);
    for(int j=0;j<5;j++){
        printf("\ndata %d : %d",j,outObj.profiles[i].data[j]);
    }
}


         for(int i=0; i<sizeof(my_data)/sizeof(int);i++)
      {
     if(my_data[i] > 0){
    index1 = my_data[i];
    myDataBinary[index1] = 1;
    printf("my data %d = %d\n",index1,myDataBinary[index1]);
}
}

       for(int i=0; i<sizeof(my_data2)/sizeof(int);i++)
{
    if(my_data2[i] > 0){
        index1 = my_data2[i];
        myDataBinary2[index1] = 1;
        printf("my data %d = %d\n",index1,myDataBinary2[index1]);
    }
}


         int sumRecievedData = 0;
         int sumMyData = 0;
         int sumMultpliedData = 0;
         float Cov =0;
         float sdMyData = 0;
         float sdRecievedData =0;
         int n = 500;
         float rho;

                for(int i=0;i<outObj.n;i++){
              index1=0;
              for (int j=0; j<outObj.profiles[i].length;j++) {

               if(outObj.profiles[i].data[j] > 0){
               index1 = outObj.profiles[i].data[j];
               recData[index1] = 1;
              printf("rec data %d = %d\n",index1,recData[index1]);
       }
     }
   }


        return 0;

    }

最佳答案

一个IP地址实际上只是一个unsigned char数组。

uchar ip[] = {127,0,0,1};

是回送地址的一种很好的表示形式。但是一个由四个字节组成的数组和一个int实际上并没有那么大。有一个异常(exception) endiannes !因此,假设我创建了代表该ip的int。天真的方法可能是:
 int ip = (127<<24)|(0<<16)|(0<<8)|(1)

当然,在小型字节序计算机上,例如x86并执行以下操作:
char *char_ip = (void*)&ip;

并反复进行此操作会产生以下结果:
1, 0, 0, 127

但是在大型Endian机器(例如PowerPC或SPARC)上,我们将拥有期望的结果,
127, 0, 0, 1

Big Endian也称为“网络字节顺序”,这是htonl中的n所代表的意思:“主机到网络长”。通过网络读取或写入整数时,经常使用这些功能。假设服务器要向客户端发送一些号码:
uint32_t important = htonl(42);
write(client, &important, sizeof important);

然后,要读取它,客户端将:
uint32_t important;
read(server, &important, sizeof important);
important = ntohl(important);

您的IP地址被翻转的原因是,因为IP地址应该按网络字节顺序排列,但是您的地址却是小端编号。 htonl -type ip上的int将为您翻转它。

关于c - htonl()和ntonhl()用于无符号字符的用法,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/6867869/

10-09 23:32