c套接字编程,一次只接收一行

时间:2012-05-03 15:27:30

标签: c sockets

我遇到的问题是我的客户端一次只收到一行。我想尝试一下接收整个流的方法。任何人都可以纠正我对接收问题的看法吗?我已经包含了客户端代码和我认为相关的服务器端代码的一部分。

#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>


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

{
    FILE *log;
    int opt;
    int count = 0;
    int sock, bytes_received;
    char send_data[2048],recv_data[2048];
    struct hostent *host;
    struct sockaddr_in server_addr;

    if(remove("client_log")) ;

    if(argc == 2)
    {
      if(strcmp(argv[1], "-v") == 0)
      {
        log = fopen("client_log", "a");
        opt = 1;
      }
      else if(strcmp(argv[1], "-V") == 0)
      {
        log = fopen("client_log", "a");
        opt = 2;
      }
    }

    }
    else
      opt = 0;

    host = gethostbyname("127.0.0.0");

    if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
        perror("Socket");
        exit(1);
    }

    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(noneya);
    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)
    {

      bytes_received=recv(sock,recv_data,2048,0);
      recv_data[bytes_received] = '\0';

      if (strcmp(recv_data , "q") == 0 || strcmp(recv_data , "Q") == 0)
      {
       close(sock);
       break;
      }

       else
      {
        if(opt != 0)
          fprintf(log, "\n Received Data: %s \n", recv_data);
        if(opt == 2)
          printf("\nReceived Data: %s \n" , recv_data);
      }

       if(count == 0)
       {
         printf("Pick an option: \n");
         printf("1 - to search by Department only\n");
         printf("2 - to search by Department and Class Number\n");
         printf("3 - to search more specifically\n");
         printf("q or Q - to Quit");
       }
       printf("\nSEND: ");
       gets(send_data);
       if(count == 0)
       {
         count = (int)send_data;
         if(count == 3)
            count++;
       }
       else if(count > 0)
         count--;

        if(opt != 0)
          fprintf(log, "\n Sending Data: %s \n", send_data);
        if(opt == 2)
          printf("\n Sending Data: %s \n" , send_data);

      if (strcmp(send_data , "q") != 0 && strcmp(send_data , "Q") != 0)
       send(sock,send_data,strlen(send_data), 0);

       else
      {
       send(sock,send_data,strlen(send_data), 0);
       close(sock);
       break;
      }

    }
return 0;
}

以下是客户方面的事情:

 case 1:

            fread(data, sizeof(sched_record), MAX_RECORD, filePointer);
            fclose(filePointer);
            char send_data[] = "Enter Department Name";
            send(connected, send_data, strlen(send_data), 0);
            bytes_received = recv(connected, recv_data, BUF, 0);
            recv_data[bytes_received] = '\0';
            strcpy(tempDept, recv_data);
            for (i=0; i<MAX_RECORD; i++){
                if ((strcmp(tempDept, data[i].Dept)==0) && tempCourse != data[i].course){
                        sprintf(send_data,"%s %d", data[i].Dept, data[i].course);
                        send(connected, send_data, strlen(send_data), 0);
                        tempCourse = data[i].course;

                }
            }
        break;

如果需要,我可以提供更多信息。我相信服务器端还可以。我用-Wall和-Wextra编译并没有收到任何警告。

2 个答案:

答案 0 :(得分:3)

您一次接收一行数据,因为它一次只发送一行。不过,它最终以这种方式出现的事实是纯粹的运气; TCP流可以由客户端TCP堆栈任意拆分和重新组合,或者通过另一端的缓冲来拆分。您需要在流中使用某种形式的显式框架 - 例如,在每行之前发送的计数,或在行尾的\n

答案 1 :(得分:1)

要添加duskwuff的答案,您在此遇到的是需要使用基于流的协议

因为TCP不是面向消息的(与UDP和其他更奇特的消息协议不同),所以您正在发送和接收单个字节流。基本上,如果你认为你的客户是一本书的作者,你的服务器是它的读者,那么在TCP世界中,所有的标点符号都被删除了:)。

作为程序员,您的工作是提供一种定义明确的方法,以便能够编码和/或嵌入一个消息/帧结束和下一个消息/帧开始的位置。使用换行符是最简单的方法,但任何基于TCP的协议都使用长度或分隔符或其组合(请参阅HTTP)。