linux下C语言多线程编程实例

  • s9_631653
    了解作者
  • 18.6KB
    文件大小
  • rar
    文件格式
  • 0
    收藏次数
  • VIP专享
    资源类型
  • 0
    下载次数
  • 2022-04-02 07:41
    上传日期
三个多线程经典代码只需要5分 转别人的,没分下载了 混点分,请多多支持
代码夹子.rar
  • 代码夹子
  • Linux下各类TCP网络服务器的实现源代码.txt
    23.8KB
  • Linux多线程编程代码(不限Linux).txt
    14.7KB
  • linux下C语言多线程编程实例.txt
    7.7KB
内容介绍
Linux下各类TCP网络服务器的实现源代码---epoll使用 来源: ChinaUnix博客  日期: 2006.07.19 14:08 (共有0条评论) 我要评论 [精华] Linux下各类TCP网络服务器的实现源代码 -------------------------------------------------------------------------------- http://www.chinaunix.net 作者:zhoulifa 发表于:2006-07-11 22:53:24 【发表评论】【查看原文】【Linux讨论区】【关闭】 大家都知道各类网络服务器程序的编写步骤,并且都知道网络服务器就两大类:循环服务和并发服务。这里附上源代码来个小结吧。 首先,循环网络服务器编程实现的步骤是这样的: 这种服务器模型是典型循环服务,如果不加上多进程/线程技术,此种服务吞吐量有限,大家都可以看到,如果前一个连接服务数据没有收发完毕后面的连接没办法处理。所以一般有多进程技术,对一个新连接启用一个新进程去处理,而监听socket继续监听。 /************关于本文档******************************************** *filename: Linux下各类TCP网络服务器的实现源代码 *purpose: 记录Linux下各类tcp服务程序源代码 *wrote by: zhoulifa( zhoulifa@163.com ) 周立发( http://zhoulifa.bokee.com ) Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言 *date time:2006-07-04 22:00:00 *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途 * 但请遵循GPL *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力 *********************************************************************/ 一个循环TCP服务源代码(因为用fork进行多进程服务了,所以这种服务现实中也有用)如下: [Copy to clipboard] [ - ] CODE: /*----------------------源代码开始--------------------------------------------*/ #include #include #include #include #include #include #include #include /********************************************************************* *filename: cycletcpserver.c *purpose: 循环tcp服务端程序 *tidied by: zhoulifa( [email=zhoulifa@163.com]zhoulifa@163.com[/email] ) 周立发( [url=http://zhoulifa.bokee.com]http://zhoulifa.bokee.com[/url] ) Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言 *date time:2006-07-04 22:00:00 *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途 * 但请遵循GPL *Thanks to: Google.com *********************************************************************/ int main(int argc, char ** argv) { int sockfd,new_fd; /* 监听socket: sock_fd,数据传输socket: new_fd */ struct sockaddr_in my_addr; /* 本机地址信息 */ struct sockaddr_in their_addr; /* 客户地址信息 */ unsigned int sin_size, myport, lisnum; if(argv[1]) myport = atoi(argv[1]); else myport = 7838; if(argv[2]) lisnum = atoi(argv[2]); else lisnum = 2; if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) { perror("socket"); exit(1); } my_addr.sin_family=PF_INET; my_addr.sin_port=htons(myport); my_addr.sin_addr.s_addr = INADDR_ANY; bzero(&(my_addr.sin_zero), 0); if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) == -1) { perror("bind"); exit(1); } if (listen(sockfd, lisnum) == -1) { perror("listen"); exit(1); } while(1) { sin_size = sizeof(struct sockaddr_in); if ((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, &sin_size)) == -1) { perror("accept"); continue; } printf("server: got connection from %s\n",inet_ntoa(their_addr.sin_addr)); if (!fork()) { /* 子进程代码段 */ if (send(new_fd, "Hello, world!\n", 14, 0) == -1) { perror("send"); close(new_fd); exit(0); } } close(new_fd); /*父进程不再需要该socket*/ waitpid(-1,NULL,WNOHANG);/*等待子进程结束,清除子进程所占用资源*/ } } /*----------------------源代码结束--------------------------------------------*/ 一个测试客户端代码如下: [Copy to clipboard] [ - ] CODE: /*----------------------源代码开始--------------------------------------------*/ #include #include #include #include #include #include #include #include #define MAXDATASIZE 100 /*每次最大数据传输量 */ /********************************************************************* *filename: cycletcpclient.c *purpose: 循环tcp客户端程序 *tidied by: zhoulifa( [email=zhoulifa@163.com]zhoulifa@163.com[/email] ) 周立发( [url=http://zhoulifa.bokee.com]http://zhoulifa.bokee.com[/url] ) Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言 *date time:2006-07-04 22:20:00 *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途 * 但请遵循GPL *Thanks to: Google.com *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力 *********************************************************************/ int main(int argc, char *argv[]) { int sockfd, numbytes; char buf[MAXDATASIZE]; struct hostent *he; struct sockaddr_in their_addr; unsigned int myport; if(argv[2]) myport = atoi(argv[2]); else myport = 7838; if (argc != 3) { fprintf(stderr,"usage: %s hostname port\n", argv[0]); exit(1); } if((he=gethostbyname(argv[1]))==NULL) { herror("gethostbyname"); exit(1); } if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) { perror("socket"); exit(1); } their_addr.sin_family=PF_INET; their_addr.sin_port=htons(myport); their_addr.sin_addr = *((struct in_addr *)he->h_addr); bzero(&(their_addr.sin_zero),0); if (connect(sockfd, (struct sockaddr *)&their_addr, sizeof(struct sockaddr)) == -1) { perror("connect"); exit(1); } if ((numbytes=recv(sockfd, buf, MAXDATASIZE, 0)) == -1) { perror("recv"); exit(1); } buf[numbytes] = 0; printf("Received: %s\n",buf); close(sockfd); return 0; } /*----------------------源代码结束--------------------------------------------*/ 用gcc cycletcpserver.c -o tcpserver和gcc cycletcpclient.c -o tcpclient分别编译上述代码后运行情况如下: 服务端运行显示: QUOTE: [email=administrator@ubuzlf:/data/example/c$]administrator@ubuzlf:/data/example/c$[/email] ./tcpserver server: got connection from 127.0.0.1 server: got connection from 127.0.0.1 server: got connection from 127.0.0.1 客户端运行显示: QUOTE: [email=administrator@ubuzlf:/data/example/c$]administrator@ubuzlf:/data/example/c$[/email] ./tcpclient 127.0.0.1 7838 Received: Hello, world! [email=administrator@ubuzlf:/data/example/c$]administrator@ubuzlf:/data/example/c$[/email] ./tcpclient 127.0.0.1 7838 Received: Hello, world! [email=administrator@ubuzlf:/data/example/c$]administrator@ubuzlf:/data/example/c$[/email] ./tcpclient 127.0.0.1 7838 Received: Hello, world! 不得不说的一个概念性问题:阻塞与非阻塞 在阻塞服务中,当服务器运行到accept语句而没有客户连接服务请求到来,那么会发生什么情况? 这时服务器就会停止在accept语句上等待连接服务请求的到来;同样,当程序运行到接收数据语句recv时,如果没有数据可以读取,则程序同样会停止在接收语句上。这种情况称为阻塞(blocking)。 但如果你希望服务器仅仅注意检查是否有客户在等待连接,有就接受连接;否则就继续做其他事情,则可以通过将 socket设置为非阻塞方式来实现:非阻塞socket在没有客户在等待时就使accept调用立即返回 。 通过设置socket为非阻塞方式,可以实现“轮询”若干socket。当企图从一个没有数据等待处理的非阻塞socket读入数据时,函数将立即返回,并且返回值置为-1,并且errno置为EWOULDBLOCK。但是这种“轮询”会使CPU处于忙等待方式,从而降低性能。考虑到这种情况,假设你希望服务器监听连接服务请求的同时从已经建立的连接读取数据,你也许会想到用一个accept语句和多个recv()语句,但是由于accept及recv都是会阻塞的,所以这个想法显然不会成功。 调用非阻塞的socket会大大地浪费系统资源。而调用select()会有效地解决这个问题,它允许你把进程本身挂起来,而同时使系统内核监听所要求的一组文件描述符的任何活动,只要确认在任何被监控的文件描述符上出现活动,select()调用将返回指示该文件描述符已准备好的信
评论
    相关推荐