webbench作为一个简单的网站压力测试工具,小巧而简单,其源码仅仅500行左右,是一个学习linux下C编程的好例子。
下载webbench:http://home.tiscali.cz/~cz210552/webbench.html,最近更新时间是04年的6月25号!!!
 解压后的webbench由下面几个文件组成:

其中有2个C源文件,一个是socket.c 另一个是webbench.c
socket.c
	
		- 
			内部仅包含一个Socket函数,如下:
		
- 
			int Socket(const char *host, int clientPort)
		
- 
			{
		
- 
			    //以host和clientPort构成一对TCP的套接字(服务器)
 
- 
			    //创建失败返回-1  成功返回一个sockt描述符
 
- 
			}
		
 
webbench.c
	
		- 
			在webbench.c源文件中,包含了下面几个函数:
		
- 
			static void alarm_handler(int signal)
		
- 
			static void usage(void)
		
- 
			void build_request(const char *url)
		
- 
			static int bench(void)
		
- 
			void benchcore(const char *host, const int port, const char *req)
		
- 
			当然还有main函数  int main(int argc, char *argv[])
		
		现在用一张图来描述一下整个webbench的工作流程:

 
	 
 
	
		- 
			这里先对全局变量做一下说明
		
- 
			/* values */
 volatile int timerexpired=0;     //根据命令行参数-t指定的测试时间判断是否超时
 int speed=0;                     //子进程成功得到服务器响应的总数
 int failed=0;                    //子进程请求失败总数
 int bytes=0;                     //读取到的字节数
 /* globals */
 int http10=1; /* 0 - http/0.9, 1 - http/1.0, 2 - http/1.1 */         //HTTP协议版本定义
 /* Allow: GET, HEAD, OPTIONS, TRACE */
 #define METHOD_GET 0
 #define METHOD_HEAD 1
 #define METHOD_OPTIONS 2
 #define METHOD_TRACE 3
 #define PROGRAM_VERSION "1.5"
 int method=METHOD_GET;           //定义HTTP请求方法GET 此外还支持OPTIONS、HEAD、TRACE方法,在main函数中用switch判断
 int clients=1;                   //默认并发数为1,也就是子进程个数 可以由命令行参数-c指定
 int force=0;                     //是否等待从服务器获取数去数据 0为获取
 int force_reload=0;              //是否使用cache  0为使用
 int proxyport=80;                //代理服务器端口 80
 char *proxyhost=NULL;            //代理服务器IP 默认为NULL
 int benchtime=30;                //测试时间  默认为30秒  可由命令行参数-t指定
 /* internal */
 int mypipe[2];                   //创建管道(半双工) 父子进程间通信,读取/写入数据
 char host[MAXHOSTNAMELEN];       //定义服务器IP
 #define REQUEST_SIZE 2048
 char request[REQUEST_SIZE];      //HTTP请求信息
 
 
由于整个webbench工具核心代码由bench和benchcore两个函数组成,下面仅分析下这两个函数的源码:
	
		- 
			/* vraci system rc error kod */
		
- 
			static int bench(void)
		
- 
			{
		
- 
			int i,j,k;
		
- 
			pid_t pid=0;
		
- 
			FILE *f;
		
- 
		
- 
			/* check avaibility of target server */
		
- 
			i=Socket(proxyhost==NULL?host:proxyhost,proxyport);   //判断是否使用代理服务器,将Socket函数的返回值进行判断
		
- 
			if(i<0) {            //错误处理 
		
- 
			 fprintf(stderr,"\nConnect to server failed. Aborting benchmark.\n");
 return 1;
 
- 
			}
		
- 
			close(i);
		
- 
			/* create pipe */
		
- 
			if(pipe(mypipe))     //创建管道及错误处理
		
- 
			{
		
- 
			perror("pipe failed.");
		
- 
			return 3;
		
- 
			}
		
- 
		
- 
			/* not needed, since we have alarm() in childrens */
		
- 
			/* wait 4 next system clock tick */
		
- 
			/*
		
- 
			cas=time(NULL);
		
- 
			while(time(NULL)==cas)
		
- 
			sched_yield();
		
- 
			*/
		
- 
		
- 
			/* fork childs */
		
- 
			for(i=0;i<clients;i++)     //根据clients大小fork出子进程,fork函数有两个返回值,在子进程中返回值为0;在父进程中返回值为新创建子进程的pid;可以根据fork的返回值,判断当前进程是父进程还是子进程
		
- 
			{
		
- 
			pid=fork();
		
- 
			if(pid <= (pid_t) 0)
		
- 
			{
		
- 
			/* child process or error*/
		
- 
			sleep(1); /* make childs faster */
		
- 
			break;
		
- 
			}
		
- 
			}
		
- 
		
- 
			if( pid< (pid_t) 0)        //错误处理,fork调用失败返回负数
		
- 
			{
		
- 
			fprintf(stderr,"problems forking worker no. %d\n",i);
		
- 
			perror("fork failed.");
		
- 
			return 3;
		
- 
			}
		
- 
		
- 
			if(pid== (pid_t) 0)        //判断pid是否为0,为0则进入子进程执行相应的代码
		
- 
			{
		
- 
			/* I am a child */
		
- 
			if(proxyhost==NULL)        //判断代理,进入benchcore函数
		
- 
			benchcore(host,proxyport,request);
		
- 
			else
		
- 
			benchcore(proxyhost,proxyport,request);
		
- 
		
- 
			/* write results to pipe */
		
- 
			f=fdopen(mypipe[1],"w");   //打开管道,子进程往管道写数据
		
- 
			if(f==NULL)                //错误处理
		
- 
			{
		
- 
			perror("open pipe for writing failed.");
		
- 
			return 3;
		
- 
			}
		
- 
			/* fprintf(stderr,"Child - %d %d\n",speed,failed); */
		
- 
			fprintf(f,"%d %d %d\n",speed,failed,bytes);    //往管道写数据
		
- 
			fclose(f);
		
- 
			return 0;
		
- 
			} else                     //判断pid是否大于0,大于0进入父进程,执行相应代码
		
- 
			{
		
- 
			f=fdopen(mypipe[0],"r");   //打开管道,父进程从管道读数据,一个管道只能进行半双工的工作(一端读,一端写)
		
- 
			if(f==NULL)
		
- 
			{
		
- 
			perror("open pipe for reading failed.");
		
- 
			return 3;
		
- 
			}
		
- 
			setvbuf(f,NULL,_IONBF,0);
		
- 
			speed=0;
		
- 
			failed=0;
		
- 
			bytes=0;
		
- 
		
- 
			while(1)
		
- 
			{
		
- 
			pid=fscanf(f,"%d %d %d",&i,&j,&k);      //从管道读数据
		
- 
			if(pid<2)
		
- 
			{
		
- 
			fprintf(stderr,"Some of our childrens died.\n");
		
- 
			break;
		
- 
			}
		
- 
			speed+=i;                  //全局计数器  speed failed bytes
		
- 
			failed+=j;
		
- 
			bytes+=k;
		
- 
			/* fprintf(stderr,"*Knock* %d %d read=%d\n",speed,failed,pid); */
		
- 
			if(--clients==0) break;    //子进程为0,数据读完后,退出循环
		
- 
			}
		
- 
			fclose(f);
		
		  //打印结果 
	
	
		- 
			printf("\nSpeed=%d pages/min, %d bytes/sec.\nRequests: %d susceed, %d failed.\n",
		
- 
			(int)((speed+failed)/(benchtime/60.0f)),
		
- 
			(int)(bytes/(float)benchtime),
		
- 
			speed,
		
- 
			failed);
		
- 
			}
		
- 
			return i;
		
- 
			}
		
 
	
		- 
			void benchcore(const char *host,const int port,const char *req)
		
- 
			{
		
- 
			int rlen;
		
- 
			char buf[1500];
		
- 
			int s,i;
		
- 
			struct sigaction sa;
		
- 
		
- 
			/* setup alarm signal handler */
		
- 
			sa.sa_handler=alarm_handler;      //加载信号处理函数
		
- 
			sa.sa_flags=0;
		
- 
			if(sigaction(SIGALRM,&sa,NULL))
		
- 
			exit(3);
		
- 
			alarm(benchtime);    //计时开始
		
- 
		
- 
			rlen=strlen(req);
		
- 
			nexttry:while(1)     //带go-to语句的while循环
		
- 
			{
		
- 
			if(timerexpired)     //超时则退出函数
		
- 
			{
		
- 
			if(failed>0)
		
- 
			{
		
- 
			/* fprintf(stderr,"Correcting failed by signal\n"); */
		
- 
			failed--;
		
- 
			}
		
- 
			return;
		
- 
			}
		
- 
			s=Socket(host,port);  //建立socket连接,获取socket描述符
		
- 
			if(s<0) { failed++;continue;}  //连接建立失败,failed++
		
- 
			if(rlen!=write(s,req,rlen)) {failed++;close(s);continue;}   //往服务器发送请求;如果请求失败,failed++,关闭当前子进程socket描述符,go-to到while循环再来一次
		
- 
			if(http10==0)         //针对HTTP0.9的处理办法
		
- 
			if(shutdown(s,1)) { failed++;close(s);continue;}
		
- 
			if(force==0)          //判断是否从服务器读取数据
		
- 
			{
		
- 
			/* read all available data from socket */
		
- 
			while(1)
		
- 
			{
		
- 
			if(timerexpired) break; //判断超时
		
- 
			i=read(s,buf,1500);      //从服务器读取数据,保存到buff数组中
		
- 
			/* fprintf(stderr,"%d\n",i); */
		
- 
			if(i<0)      //读取数据失败的处理,返回while循环开始处,重新来一次
		
- 
			{
		
- 
			failed++;
		
- 
			close(s);
		
- 
			goto nexttry;
		
- 
			}
		
- 
			else
		
- 
			if(i==0) break;   //读取成功后,bytes统计数据大小
		
- 
			else
		
- 
			bytes+=i;
		
- 
			}
		
- 
			}
		
- 
			if(close(s)) {failed++;continue;}
		
- 
			speed++;
		
- 
			}
		
- 
			}
		
 
由于本人对Linux下的C编程仅仅停留在粗略理解《linux C编程实战》一书的基础上,本文如有错误,还请指正
webbench源码浅析
原文:http://blog.chinaunix.net/uid-28841896-id-4773295.html