一、使用fopen来创建文件
当tftp的RRQ消息发送成功后,正常情况下服务器端就会回发第一个Data数据包,这时,客户端需要做的是,在本地新建一个文件,以便将从服务器端接收的数据写入到此文件中。现在,我们暂时使用fopen来创建文件。
对于fopen()函数,简单介绍一下:
函数原型:FILE * fopen(const char * path,const char * mode);
返回值:文件顺利打开后,指向该流的文件指针就会被返回。如果文件打开失败则返回NULL,并把错误代码存在errno中。
一般而言,打开文件后会做一些文件读取或写入的动作,若打开文件失败,接下来的读写动作也无法顺利进行,所以一般在fopen()后作错误判断及处理。
参数说明:
参数path字符串包含欲打开的文件路径及文件名,参数mode字符串则代表着流形态。
mode有下列几种形态字符串,只列举部分 :
“r” 以只读方式打开文件,该文件必须存在。
“r+” 以可读写方式打开文件,该文件必须存在。
“rb+” 读写打开一个二进制文件,允许读写数据,文件必须存在。
“w” 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。
“w+” 打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。
“a” 以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。(EOF符保留)
“a+“ 以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。 (原来的EOF符不保留)
“wb” 只写打开或新建一个二进制文件;只允许写数据。
具体使用:
/*生成需要接收的文件*/
FILE *fp = fopen("请写完整文件名","wb");
if(NULL == fp)
{
perror("fopen()");
return 1;
}
二、定义sockaddr_in结构变量,用于保存服务端的IP及端口号,以便后续的通信使用。
struct sockaddr_in peeraddr;
memset(&peeraddr, 0, sizeof(peeraddr));
三、进入while循环,准备接收数据:
其中,ioctl的作用为从sockfd对应的连接中读取数据,FIONREAD参数指明获取接收缓存区中的字节数,并把读取的字节数的值赋给bufferlen。
while(1)
{
int bufferlen = 0;
/*在此阻塞,等待数据的到来,使用此函数可以知道有多少数据到达接收缓冲器区,
然后根据接收的数据长度使用malloc()函数来进行动态内存分配*/
while(bufferlen == 0)
{
ioctl(sockfd, FIONREAD, &bufferlen);
}
printf("DBG:bufferlen = %d\n", bufferlen);
void *buffer = malloc(bufferlen); /*根据到达的数据的大小,动态分配内存*/
int recvLen = sizeof(peeraddr);
int recvlen