求助:HTTP协议的C语言编程实现实例

dasenlin1988 2011-09-05 08:45:40

大家好,最近需要做一个HTTP多线程编程的软件,先从网站上下载了如下的代码,想先研究透彻之后再写自己的多线程代码,在linux系统下编译后 执行./httpclient http://dl.ludashi.com/ludashi/ludashisetup.zip 后面的地址是我想下载的文件 可是运行的时候总是停在如下的地方 请大家帮忙分析下是什么原因啊 谢啦 我是个菜鸟
root@ubuntu:/home/zzf/9-4# ./clienthttp. http://dl.ludashi.com/ludashi/ludashisetup.zip
parameter.1 is: http://dl.ludashi.com/ludashi/ludashisetup.zip
webhost:dl.ludashi.com
hostfile:ludashi/ludashisetup.zip
portnumber:80

GET /ludashi/ludashisetup.zip HTTP/1.1

Accept-Language: zh-cn
User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)
Host: dl.ludashi.com:80
Connection: Close


211 bytes send OK!

The following is the response header:

下面是源程序
/******* http客户端程序 httpclient.c ************/

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <errno.h>
#include <unistd.h>
#include <netinet/in.h>
#include <limits.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <ctype.h>
//////////////////////////////httpclient.c 开始///////////////////////////////////////////
/********************************************

功能:搜索字符串右边起的第一个匹配字符

********************************************/
char* Rstrchr(char* s, char x)
{
int i = strlen(s);
if(!(*s))
{
return 0;
}
while(s[i-1])
{
if(strchr(s+(i-1), x))
{
return (s+(i-1));
}
else
{
i--;
}
}
return 0;
}

/********************************************

功能:把字符串转换为全小写

********************************************/
void ToLowerCase(char* s)
{
while(*s && *s!='\0' )
{
*s=tolower(*s++);
}
*s = '\0';
}

/**************************************************************
功能:从字符串src中分析出网站地址和端口,并得到用户要下载的文件

***************************************************************/
void GetHost(char* src, char* web, char* file, int* port)
{
char* pA;
char* pB;
memset(web, 0, sizeof(web));
memset(file, 0, sizeof(file));
*port = 0;
if(!(*src))
{
return;
}
pA = src;
if(!strncmp(pA, "http://", strlen("http://")))
{
pA = src+strlen("http://");
}
else if(!strncmp(pA, "https://", strlen( "https://")))
{
pA = src+strlen( "https://");
}
pB = strchr(pA, '/');
if(pB)
{
memcpy(web, pA, strlen(pA)-strlen(pB));
if(pB+1)
{
memcpy(file, pB+1, strlen(pB)-1);
file[strlen(pB)-1] = 0;
}
}
else
{
memcpy(web, pA, strlen(pA));
}
if(pB)
{
web[strlen(pA) - strlen(pB)] = 0;
}
else
{
web[strlen(pA)] = 0;
}
pA = strchr(web, ':');
if(pA)
{
*port = atoi(pA + 1);
}
else
{
*port = 80;
}
}

/*********************************************************************

*filename: httpclient.c

*purpose: HTTP协议客户端程序,可以用来下载网页

*********************************************************************/
int main(int argc, char *argv[])
{
int sockfd = 0;
char buffer[1024] = "";
struct sockaddr_in server_addr;
struct hostent *host;
int portnumber = 0;
int nbytes = 0;
char host_addr[256] = "";
char host_file[1024] = "";
FILE *fp;
char request[1024] = "";
int send = 0;
int totalsend = 0;
int i = 0;
char *pt;
char local_file[256]="/home/zzf/baidu.zip";//重命名接收到的文件

if(argc!=2)
{
fprintf(stderr, "Usage:%s web-address\a\n ",argv[0]);
exit(1);
}
printf( "parameter.1 is: %s\n ", argv[1]);
//ToLowerCase(argv[1]);/*将参数转换为全小写*/
//printf( "lowercase parameter.1 is: %s\n ", argv[1]);
GetHost(argv[1], host_addr, host_file, &portnumber);/*分析网址、端口、文件名等*/
printf( "webhost:%s\n ", host_addr);
printf( "hostfile:%s\n ", host_file);
printf( "portnumber:%d\n\n ", portnumber);
if((host=gethostbyname(host_addr)) == NULL)/*取得主机IP地址*/
{
fprintf(stderr, "Gethostname error, %s\n ", strerror(errno));
exit(1);
}

/* 客户程序开始建立 sockfd描述符 */
if((sockfd=socket(AF_INET,SOCK_STREAM,0)) == -1)/*建立SOCKET连接*/
{
fprintf(stderr, "Socket Error:%s\a\n ",strerror(errno));
exit(1);
}
/* 客户程序填充服务端的资料 */
bzero(&server_addr,sizeof(server_addr));
server_addr.sin_family=AF_INET;
server_addr.sin_port=htons(portnumber);
server_addr.sin_addr=*((struct in_addr*)host->h_addr);
/* 客户程序发起连接请求 */

if(connect(sockfd, (struct sockaddr*)(&server_addr), sizeof(struct sockaddr)) == -1)/*连接网站*/
{
fprintf(stderr, "Connect Error:%s\a\n ",strerror(errno));
exit(1);
}
sprintf(request, "GET /%s HTTP/1.1\r\n\r\nAccept-Language: zh-cn\r\n\
User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)\r\n\
Host: %s:%d\r\nConnection: Close\r\n\r\n ", host_file, host_addr, portnumber);
printf( "%s\n", request);/*准备request,将要发送给主机*/

/*取得真实的文件名*/
if(host_file && *host_file)
{
pt = Rstrchr(host_file, '/');
}
else
{
pt = 0;
}
/*发送http请求request*/
send = 0;
totalsend = 0;
nbytes=strlen(request);
while(totalsend < nbytes)
{
send = write(sockfd, request+totalsend, nbytes-totalsend);
if(send == -1)
{
printf( "send error!%s\n ", strerror(errno));
exit(0);
}
totalsend += send;
printf("%d bytes send OK!\n ", totalsend);
}

fp = fopen(local_file, "a");
if(!fp)
{
printf("create file error! %s\n", strerror(errno));
return 0;
}
printf("\nThe following is the response header:\n");
i=0;
/* 连接成功了,接收http响应,response */
while((nbytes=read(sockfd,buffer,1))==1)
{
if(i < 4)
{

if(buffer[0] == '\r' || buffer[0] == '\n')
i++;
else i = 0;
printf("%c", buffer[0]);/*把http头信息打印在屏幕上*/
}
else
{
fwrite(buffer, 1, 1, fp);/*将http主体信息写入文件*/
i++;
if(i%1024 == 0) fflush(fp);/*每1K时存盘一次*/
}
}
// printf("\n\nresponse = %s\n", buffer);
/* 结束通讯 */
fclose(fp);
close(sockfd);
exit(0);
}

//////////////////////////////httpclient.c 结束///////////////////////////////////////////
...全文
299 3 打赏 收藏 转发到动态 举报
写回复
用AI写文章
3 条回复
切换为时间正序
请发表友善的回复…
发表回复
luciferisnotsatan 2011-09-06
  • 打赏
  • 举报
回复
linux下可以使用封装好的libcurl
windows下用wininet
Ansny 2011-09-06
  • 打赏
  • 举报
回复
发现一个疑问:
sprintf(request, "GET /%s HTTP/1.1\r\n\r\nAccept-Language: zh-cn\r\n\

这里出现了 \r\n\r\n 应该是 \r\n

建议使用 HTTP 库。
敬请关注 http://code.google.com/p/cutehttpd/ 本人搞的项目。
或者参考 http://code.google.com/p/mongoose/ 项目
还有 MiniWebSvr
wintree 2011-09-05
  • 打赏
  • 举报
回复
好长的了~~你打个断点,看看有什么错误,你贴出来啊~~~~看看错误的~~~


帮你顶了!~~接分~

69,371

社区成员

发帖
与我相关
我的任务
社区描述
C语言相关问题讨论
社区管理员
  • C语言
  • 花神庙码农
  • 架构师李肯
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧