-1

我编写了一个简单的服务器来将文件传输到客户端。这里是源代码。

//server.c
#include "general.h"
#define LISTENQ 10
#define BUFSIZE 1024
#define FILENAME "List"

void sendlist(int sockfd) {
    int fd = open(FILENAME, O_RDONLY);
    char readbuf[BUFSIZE];
    ssize_t n;

    if(fd == -1)
            exit(1);

    for(;;) {
            if((n = read(fd, readbuf, BUFSIZE)) > 0)
                    write(sockfd, readbuf, BUFSIZE);
            else
                    break;
    }

    close(fd);
    close(sockfd);
}

int main(int argc, char ** argv) {
    int listenfd, connfd;
    pid_t childpid;
    socklen_t len;
    struct sockaddr_in cliaddr, servaddr;

    listenfd = socket(AF_INET, SOCK_STREAM, 0);

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

    bind(listenfd, (struct sockaddr_in *)&servaddr, sizeof(servaddr));
    listen(listenfd,LISTENQ);

    for(;;) {
            len = sizeof(cliaddr);
            connfd = accept(listenfd, (struct sockaddr_in *)&cliaddr, &len);
            if((childpid = fork()) == 0) {
                    close(listenfd);
                    sendlist(connfd);
                    exit(0);
            }
            close(connfd);
    }
}



//client.c
#include "general.h"

#define MAXLINE 1024
#define FILENAME "List2"

void getlist(int sockfd) {
    int fd = creat(FILENAME, S_IRUSR | S_IWUSR);
    char buf[MAXLINE];
    ssize_t n;

    if(fd == -1)
            exit(2);

    for(;;) {
            if((n = read(sockfd, buf, MAXLINE)) > 0)
                    write(fd, buf, MAXLINE);
            else
                    break;
    }

    close(fd);
}

int main(int argc, char ** argv) {
    int sockfd;
    struct sockaddr_in servaddr;

    if(argc != 2)
            exit(1);

    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(SERV_PORT);
    inet_pton(AF_INET, argv[1], &servaddr.sin_addr);

    connect(sockfd, (struct sockaddr_in *)&servaddr, sizeof(servaddr));

    getlist(sockfd);

    exit(0);
}

我的问题是:文件 List 和 List2 不一样。例如文件 List 的内容是:test 文件 List2 的内容不仅包含test,而且还包含很多不可读的代码,如下所示:

^@^@^@^@^@^@^@^@^@<8c>]úUü^?^@^@ ^G^\Vü^?^@^@^A^@^@^

我认为原因是读写功能。我不应该使用它们吗?谢谢

4

3 回答 3

4

您正在将整个缓冲区写入文件,而只有第一个n字节有效。代替:

for(;;) {
        if((n = read(sockfd, buf, MAXLINE)) > 0)
                write(fd, buf, MAXLINE);

和:

for(;;) {
        if((n = read(sockfd, buf, MAXLINE)) > 0)
               write(fd, buf, n);

并相应地发生所有类似事件。

于 2013-05-12T14:13:40.600 回答
1
  1. 我相信你需要先 memset 缓冲区
  2. 仅发送您实际阅读的内容,而不是您拥有的所有内容
  3. 只写你收到的而不是缓冲区大小

喜欢

 for(;;) {
     memset(readbuf , 0, BUFSIZE);   // Clear Memory here
     if((n = read(fd, readbuf, BUFSIZE)) > 0)
                write(sockfd, readbuf, n); // Send what you read
        else
                break;
}



 // similarly While reading
 for(;;) {

        memset(buf , 0 , MAXLINE); // again do a memset 

        if((n = read(sockfd, buf, MAXLINE)) > 0)
                write(fd, buf, n); // Write What you read
        else
                break;
}
于 2013-05-12T14:24:12.430 回答
0

Read() 和 write() 可以进行部分读/写。这由它们的返回值表示,该值是成功传输的字节数,它可能与提供给它们的第三个参数不同。

while(1) {
    int n;
    unsigned todo, done;
    n = read(fd, readbuf, BUFSIZE);
    if (n <= 0) break;

    for(todo=n,done=0; done < todo; done += n ) {
        n = write(sockfd, readbuf+done, todo - done);
        if (n <= 0) return FAILURE;
        }
    }
于 2013-05-12T14:22:09.600 回答