要求实现一个简单的定时器,该程序要求的输入格式为:
N prompt message
其中N表示时间,prompt message表示提示信息。即过了N秒之后要在终端上显示出提示信息。一共用了三种方式实现:(1)最普通的方式,阻塞I/0+同步。(2)使用进程实现。(3)使用线程实现。
/*alarm.c*/
#include "../error.h"
#include<time.h>
int main(int argc,char *argv[])
{
int seconds;
char line[128];
char message[64];
while(NULL != fgets(line,sizeof(line),stdin))
{
if( 2 > sscanf(line,"%d %64[^\n]",&seconds,message))
fprintf(stderr,"Bad Command\n");
else
{
sleep(seconds);
fprintf(stdout,"%s\n",message);
}
}
return 0;
}
评注:
1.常量和函数比较的时候,将常量放在前面可以防止将==写成=,因为编译器会报错。
2.sscanf为从一个字符缓冲区中格式化读取数据。注意字符缓冲区放在第一个参数位置。使用%N可以控制读取字符的数量,中括号的作用和正则表达式中的作用一样,将不读取的字符排除掉或者仅读取中括号内指定的字符集,成功时,返回读取的参数的个数。
3.使用fgets+sscanf,简化了从命令行中提取参数问题。
3.fgets从指定的文件描述符中读取一行字符串,失败时返回NULL。
4.对于每个函数都添加了判断语句,特别有助于调试。
/*alarm_fork.c*/
#include <sys/types.h>
#include<wait.h>
pid_t pid;
if(2 > sscanf(line,"%d %64[^\n]",&seconds,message))
fprintf(stdout,"Bad command.\n");
continue;
pid = fork();
if((pid_t)-1 == pid)
{
error_abort("fork error...");
}
else if((pid_t)0 == pid)
sleep(seconds);
fprintf(stdout,"%s\n",message);
exit(0);
else
do
{
pid = waitpid((pid_t)-1,NULL,WNOHANG);
if((pid_t)-1 == pid)
{
error_abort("waitpid error...");
}
}while((pid_t)0 != pid);
1.pid_t这个类型在sys/types.h头文件中。
2.比较的时候要进行强制类型转换,比如判断进程id是不是为0,就要使用(pid_t)0 == pid这样的判断语句。
3.waitpid的第一个参数设置为-1,第三个参数设置为WNOHANG,表示非阻塞等待任何一个子进程。如果发现一个子进程返回,立即再判断是否还有其他进程返回。可以迅速的释放不再使用的资源。
/*alarm_thread.c*/
#include <pthread.h>
#include <time.h>
typedef struct alarm_tag
}alarm_t,*alarm_p;
void * alarm_thread(void * arg)
int status;
status = pthread_detach(pthread_self());
if(0 != status)
err_abort("detaching thread failure...",status);
alarm_p alarm = (alarm_p) arg;
sleep(alarm->seconds);
fprintf(stdout,"%s\n",alarm->message);
pthread_t thread;
alarm_p alarm;
alarm = (alarm_p) malloc(sizeof(alarm_t));
if( NULL == alarm)
error_abort("Allocating alarm failure...");
if(2 > sscanf(line,"%d %64[^\n]",&alarm->seconds,alarm->message))
fprintf(stdout,"%s\n","Bad command");
free(alarm);
status = pthread_create(&thread,NULL,alarm_thread,(void *)alarm);
if(0 != status)
err_abort("creating thread failure...",status);
1.Pthreads相关的类型,接口原型,常量都在pthread.h这个头文件中,编译的时候要加 -lpthread.
2.由于线程的参数只有一个,所以要将传给线程的所有参数封装到一个结构体中。
使用普通方式,多进程,多线程的比较
alarm一次只能发出一个定时请求。如果发出一个10分钟的请求,那么必须要等十分钟才能发出下一个请求。多进程解决了这个同步问题,但是在一个系统中,一个用户能够启动的进程的数量是非常有限的,多线程受到这个影响要小得多。
多线程的几个好处
(1)发掘多核计算潜力
(2)发掘程序自身的并发性
(3)模块式的编程模型,可以更加清晰的表达不同事件之间的关系
本文转自hipercomer 51CTO博客,原文链接:http://blog.51cto.com/hipercomer/908959