Chinaunix首页 | 论坛 | 博客
  • 博客访问: 95365
  • 博文数量: 34
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 12
  • 用 户 组: 普通用户
  • 注册时间: 2013-12-18 15:36
文章分类

全部博文(34)

文章存档

2014年(8)

2013年(26)

我的朋友

分类: C/C++

2013-12-25 15:02:00

原文地址:Linux下http协议实现 作者:chenshko

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <sys/types.h>
  5. #include <sys/socket.h>
  6. #include <errno.h>
  7. #include <unistd.h>
  8. #include <netinet/in.h>
  9. #include <limits.h>
  10. #include <netdb.h>
  11. #include <arpa/inet.h>
  12. #include <ctype.h>

  13. int main(int argc, char *argv[])
  14. {
  15. int sockfd;
  16. char buffer[1024];
  17. struct sockaddr_in server_addr;
  18. struct hostent *host;
  19. int portnumber,nbytes;
  20. char host_addr[256];
  21. char host_file[1024];
  22. char local_file[256];
  23. FILE * fp;
  24. char request[1024];
  25. int send, totalsend;
  26. int i;
  27. char * pt;

  28. if(argc!=2)
  29. {
  30.     fprintf(stderr,"Usage:%s web-address\a\n",argv[0]);
  31.     exit(1);
  32. }
  33. portnumber=80;
  34. strcpy(host_addr,argv[1]);
  35. if((host=gethostbyname(argv[1]))==NULL)/*取得主机IP地址*/
  36. {
  37.     fprintf(stderr,"Gethostname error, %s\n", strerror(errno));
  38.     exit(1);
  39. }
  40. if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1)/*建立SOCKET连接*/
  41. {
  42.     fprintf(stderr,"Socket Error:%s\a\n",strerror(errno));
  43.     exit(1);
  44. }
  45. /* 客户程序填充服务端的资料 */
  46. bzero(&server_addr,sizeof(server_addr));
  47. server_addr.sin_family=AF_INET;
  48. server_addr.sin_port=htons(portnumber);
  49. server_addr.sin_addr=*((struct in_addr *)host->h_addr);

  50. /* 客户程序发起连接请求 */
  51. if(connect(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)/*连接网站*/
  52. {
  53.     fprintf(stderr,"Connect Error:%s\a\n",strerror(errno));
  54.     exit(1);
  55. }

  56. sprintf(request, "GET /%s HTTP/1.1\r\nAccept: */*\r\nAccept-Language: zh-cn\r\n\
  57. User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)\r\n\
  58. Host: %s:%d\r\nConnection: Close\r\n\r\n", host_file, host_addr, portnumber);
  59. printf("%s", request);/*准备request,将要发送给主机*/

  60. /*取得真实的文件名*/
  61. strcpy(local_file, "index.html");
  62. /*发送http请求request*/
  63. send = 0;totalsend = 0;
  64. nbytes=strlen(request);
  65. while(totalsend < nbytes) {
  66.     send = write(sockfd, request + totalsend, nbytes - totalsend);
  67.     if(send==-1) {printf("send error!%s\n", strerror(errno));exit(0);}
  68.     totalsend+=send;
  69.     printf("%d bytes send OK!\n", totalsend);
  70. }

  71. fp = fopen(local_file, "a");
  72. if(!fp) {
  73.     printf("create file error! %s\n", strerror(errno));
  74.     return 0;
  75. }
  76. printf("\nThe following is the response header:\n");
  77. i=0;
  78. /* 连接成功了,接收http响应,response */
  79. while((nbytes=read(sockfd,buffer,1))==1)
  80. {
  81.     if(i < 4) {
  82.       if(buffer[0] == '\r' || buffer[0] == '\n') i++;
  83.       else i = 0;
  84.       printf("%c", buffer[0]);/*把http头信息打印在屏幕上*/
  85.     }
  86.     else {
  87.       fwrite(buffer, 1, 1, fp);/*将http主体信息写入文件*/
  88.       i++;
  89.       if(i%1024 == 0) fflush(fp);/*每1K时存盘一次*/
  90.     }
  91. }
  92. fclose(fp);
  93. /* 结束通讯 */
  94. close(sockfd);
  95. exit(0);
  96. }

 

1    struct hostent *gethostbyname(const char *name);

     这个函数完成进行域名和IP地址的转换,返回的为:

      struct hostent {
      char *h_name;                      /* 主机的官方域名 */
      char **h_aliases;                  /* 一个以NULL结尾的主机别名数组 */
      int h_addrtype;                     /* 返回的地址类型,在Internet环境下为AF-INET */
     int h_length;                         /*地址的字节长度 */
     char **h_addr_list;                /* 一个以0结尾的数组,包含该主机的所有地址*/  
      };  
     #define h_addr h_addr_list[0]           /*在h-addr-list中的第一个地址*/

    我们一般用的也就第一个地址

2   接下去就是SOCKET 的建立,绑定,连接,我们想要连接,上面得到的ip地址是不行的,我们要使用服务器的地址,具体数据结构如下:

   struct sockaddr_in {
      short int sin_family;             /* 地址族 */
      unsigned short int sin_port;      /* 端口号 */
      struct in_addr sin_addr;          /* IP地址 */ 上面得到的地址
      unsigned char sin_zero[8];        /* 填充0 以保持与struct sockaddr同样大小 */
      };

   这里也提下 struct sockaddr 这个是描述sock 地址信息的,和上面的结构大小一样,可以相互转换

   struct sockaddr {
       unsigned short sa_family; /* 地址族, AF_xxx */ 
       char sa_data[14];         /* 14 字节的协议地址 */
   }

3 我们和服务器连上之后,就可以向服务器发送请求了

     write(sockfd, char *, size);

内容就是:GET /%s HTTP/1.1\r\nAccept: */*\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

   具体是由其协议定的,我也还不是很清楚

4 服务器响应,就会发来信息头+ 实际页面的信息, 这个中间是有4个("\r"或"\n")进行分开的。


 

阅读(1303) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~