linux socket编程很好的学习文档

源代码在线查看: linux网络编程一步一步学-https客户端程序示例(20).txt

软件大小: 94 K
上传用户: lwsnsfp17
关键词: socket linux 编程 文档
下载地址: 免注册下载 普通下载 VIP

相关代码

				Linux网络编程一步一步学-HTTPS客户端程序示例
				关键词: Linux    网络    加密通讯    SSL_read    SSL_write                                           
				源代码如下:
				
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				#include 
				
				#define DEBUG 1
				
				/********************************************
				功能:搜索字符串右边起的第一个匹配字符
				********************************************/
				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;
				}
				
				/**************************************************************
				功能:从字符串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 = 443;
				}
				
				/************关于本文档********************************************
				*filename: https-client.c
				*purpose: 演示HTTPS客户端编程方法
				*wrote by: zhoulifa(zhoulifa@163.com) 周立发(http://zhoulifa.bokee.com)
				Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言
				*date time:2007-01-30 20:06
				*Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
				* 但请遵循GPL
				*Thanks to:Google
				*Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
				* 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
				*********************************************************************/
				
				int main(int argc, char *argv[])
				{
				    int sockfd, ret;
				    char buffer[1024];
				    struct sockaddr_in server_addr;
				    struct hostent *host;
				    int portnumber, nbytes;
				    char host_addr[256];
				    char host_file[1024];
				    char local_file[256];
				    FILE *fp;
				    char request[1024];
				    int send, totalsend;
				    int i;
				    char *pt;
				    SSL *ssl;
				    SSL_CTX *ctx;
				
				    if (argc != 2) {
				        if (DEBUG)
				            fprintf(stderr, "Usage:%s webpage-address\a\n", argv[0]);
				        exit(1);
				    }
				    if (DEBUG)
				        printf("parameter.1 is: %s\n", argv[1]);
				
				    GetHost(argv[1], host_addr, host_file, &portnumber);        /*分析网址、端口、文件名等 */
				    if (DEBUG)
				        printf("webhost:%s\n", host_addr);
				    if (DEBUG)
				        printf("hostfile:%s\n", host_file);
				    if (DEBUG)
				        printf("portnumber:%d\n\n", portnumber);
				
				    if ((host = gethostbyname(host_addr)) == NULL) {        /*取得主机IP地址 */
				        if (DEBUG)
				            fprintf(stderr, "Gethostname error, %s\n", strerror(errno));
				        exit(1);
				    }
				
				    /* 客户程序开始建立 sockfd描述符 */
				    if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {        /*建立SOCKET连接 */
				        if (DEBUG)
				            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) {        /*连接网站 */
				        if (DEBUG)
				            fprintf(stderr, "Connect Error:%s\a\n", strerror(errno));
				        exit(1);
				    }
				
				    /* SSL初始化 */
				    SSL_library_init();
				    SSL_load_error_strings();
				    ctx = SSL_CTX_new(SSLv23_client_method());
				    if (ctx == NULL) {
				        ERR_print_errors_fp(stderr);
				        exit(1);
				    }
				
				    ssl = SSL_new(ctx);
				    if (ssl == NULL) {
				        ERR_print_errors_fp(stderr);
				        exit(1);
				    }
				
				    /* 把socket和SSL关联 */
				    ret = SSL_set_fd(ssl, sockfd);
				    if (ret == 0) {
				        ERR_print_errors_fp(stderr);
				        exit(1);
				    }
				
				    RAND_poll();
				    while (RAND_status() == 0) {
				        unsigned short rand_ret = rand() % 65536;
				        RAND_seed(&rand_ret, sizeof(rand_ret));
				    }
				
				    ret = SSL_connect(ssl);
				    if (ret != 1) {
				        ERR_print_errors_fp(stderr);
				        exit(1);
				    }
				
				    sprintf(request, "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", host_file, host_addr,
				            portnumber);
				    if (DEBUG)
				        printf("%s", request);        /*准备request,将要发送给主机 */
				
				    /*取得真实的文件名 */
				    if (host_file && *host_file)
				        pt = Rstrchr(host_file, '/');
				    else
				        pt = 0;
				
				    memset(local_file, 0, sizeof(local_file));
				    if (pt && *pt) {
				        if ((pt + 1) && *(pt + 1))
				            strcpy(local_file, pt + 1);
				        else
				            memcpy(local_file, host_file, strlen(host_file) - 1);
				    } else if (host_file && *host_file)
				        strcpy(local_file, host_file);
				    else
				        strcpy(local_file, "index.html");
				    if (DEBUG)
				        printf("local filename to write:%s\n\n", local_file);
				
				    /*发送https请求request */
				    send = 0;
				    totalsend = 0;
				    nbytes = strlen(request);
				    while (totalsend < nbytes) {
				        send = SSL_write(ssl, request + totalsend, nbytes - totalsend);
				        if (send == -1) {
				            if (DEBUG)
				                ERR_print_errors_fp(stderr);
				            exit(0);
				        }
				        totalsend += send;
				        if (DEBUG)
				            printf("%d bytes send OK!\n", totalsend);
				    }
				
				    fp = fopen(local_file, "a");
				    if (!fp) {
				        if (DEBUG)
				            printf("create file error! %s\n", strerror(errno));
				        return 0;
				    }
				    if (DEBUG)
				        printf("\nThe following is the response header:\n");
				    i = 0;
				    /* 连接成功了,接收https响应,response */
				    while ((nbytes = SSL_read(ssl, buffer, 1)) == 1) {
				        if (i < 4) {
				            if (buffer[0] == '\r' || buffer[0] == '\n')
				                i++;
				            else
				                i = 0;
				            if (DEBUG)
				                printf("%c", buffer[0]);        /*把https头信息打印在屏幕上 */
				        } else {
				            fwrite(buffer, 1, 1, fp);        /*将https主体信息写入文件 */
				            i++;
				            if (i % 1024 == 0)
				                fflush(fp);        /*每1K时存盘一次 */
				        }
				    }
				    fclose(fp);
				    /* 结束通讯 */
				    ret = SSL_shutdown(ssl);
				    if (ret != 1) {
				        ERR_print_errors_fp(stderr);
				        exit(1);
				    }
				    close(sockfd);
				    SSL_free(ssl);
				    SSL_CTX_free(ctx);
				    ERR_free_strings();
				    exit(0);
				}
				 
				
				编译此程序用下列命令:
				gcc -Wall https-client.c -lssl -o httpsclient
				
				运行此程序来取得HTTPS服务器上的页面,比如:
				./httpsclient https://127.0.0.1/test.html
				
				关键之处在于建立socket之后的SSL相关初始化以及中间的recv/send用SSL_read和SSL_write代替,最后记得释放SSL资源即可。
				可以对比之前的文章来发现异同: 
				
				
				HTTP协议的C语言编程实现实例
				
				
				
				
				
				【作者: 周立发】【访问统计:2542】【2007年01月30日 星期二 20:22】【注册】【打印】 
				
				Google Adsense
				Burn-in Sockets by Aries
				Std. & Custom IC Burn-in Sockets For BGA, DIP, PGA, Plcc, & others 
				www.arieselec.com/
				 
				
				搜索
				            
				Trackback
				你可以使用这个链接引用该篇文章 http://publishblog.blogchina.com/blog/tb.b?diaryID=6079174 
				
				来自租房网的租房网的引用:
				
				
				租房网
				回复
				 - 评论人:michael   2007-12-04 12:43:41      
				释放ssl资源的时候,ret = SSL_shutdown(ssl); 
				但跟踪调试的时候,ret并不为1,这样的话,是不是就是没有释放ssl资源呀
				
				 
				 - 评论人:周立发   2007-03-01 18:35:41      
				这个得研究研究,据说是为了产生随机数,通讯双方在这有个协调过程,如果随机数不对通讯会失败。查好再回复
				
				 
				 - 评论人:xiaozhao   2007-03-01 09:56:33      
				一些不明白的地方,周老师能否给讲解一下,谢了! 
				RAND_poll(); 
				while (RAND_status() == 0) { 
				unsigned short rand_ret = rand() % 65536; 
				RAND_seed(&rand_ret, sizeof(rand_ret)); 
				} 
				以上代码的作用是什么?
				
				 
							

相关资源