目录
1 信号(signal)
2 设置信号响应方式 – signal
3 设置定时器
4 示例
问题:
getch()阻塞获取键盘按键输入, 怎么操作才能不影响小鸟下落和管道移动?
getch如果阻塞,下面的程序都是无法执行。通过信号机制方式实现。
1 信号(signal)
在Linux中,软中断信号(signal,简称为信号)是在软件层次上对中断的一种模拟,用来通知进程发生了异步事件。内核可以因为内部事件而给进程发送信号,通知进程发生了某个事件。
信号响应的方式:
1.忽略信号,即对信号不做任何处理;
2.捕捉信号,即信号发生时执行用户自定义的信号处理函数。
3.执行缺省操作,Linux对每种信号都规定了默认操作。
信号的检测与处理流程图
2 设置信号响应方式 – signal
#include <unistd.h>
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
成功时返回原先的信号处理函数,失败时返回SIG_ERR
signum:指明了所要处理的信号类型
handler:描述了与信号关联的动作
SIG_DFL代表缺省方式; SIG_IGN 代表忽略信号;
指定的信号处理函数代表捕捉方式
示例
// 头文件省略
void handler (int signo) {
printf(“HELLO!\n”);
}
int main() {
signal(SIGINT, handler);
while ( 1 ) ;
return 0;
}
3 设置定时器
struct itimerval {
struct timeval it_interval; /* 计时器重新启动的间歇值 */
struct timeval it_value; /* 计时器安装后首次启动的初 }; 始值,之后就没有用 */
struct timeval {
long tv_sec; /* 秒 */
long tv_usec; /* 微妙*/
};
计时器的实现
int setitimer(int which, const struct itimerval *value,
struct itimerval *ovalue)
参数: which:间歇计时器类型,
ITIMER_REAL //数值为0,发送的信号是SIGALRM。
struct itimerval *value:将value指向的结构体设为计时器的当前值,
struct itimerval *ovalue:保存计时器原有值。一般设置为NULL。
返回值: 成功返回0。失败返回-1。
4 示例
#include <stdio.h>
#include <curses.h>
#include <signal.h>
#include <sys/time.h>
int x=10,y=10;
int a=5,b=10;
void handler(int sig)
{
move(a,b);
addch('B');
refresh();
b++;
}
int main(int argc, const char *argv[])
{
char ch;
initscr();//进入curses模式
curs_set(0);
noecho();//禁止字符显示
keypad(stdscr,1);//允许使用功能键
start_color();//启动颜色机制
init_pair(1,COLOR_WHITE, COLOR_RED);
init_pair(2,COLOR_GREEN,COLOR_WHITE);
signal(SIGALRM, handler);
/*设置定时时间*/
struct itimerval timer;
timer.it_value.tv_sec = 3;//首次启动定时时间
timer.it_value.tv_usec = 0;
timer.it_interval.tv_sec = 1;//之后每次的定时时间
timer.it_interval.tv_usec = 0;
/*启动定时*/
setitimer(ITIMER_REAL, &timer, NULL);
while(1){
ch = getch();
if(ch == 'Q')
{
attron(COLOR_PAIR(1));
move(x,y);
addch('A');
refresh();
y++;
attroff(COLOR_PAIR(1));
}
}
while(1);
endwin();//退出curses模式
return 0;
}
5 练习
使用信号机制实现每隔1s打印字符串“Flappy bird”,提交代码。
方式1
#include <signal.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/time.h>
void handle(int sig)
{
printf("Flappy bird\n");
}
int main(int argc,char *argv[])
{
struct sigaction act;
struct itimerval timevalue;
int ret;
act.sa_handler = handle;
act.sa_flags = 0;
sigemptyset(&act.sa_mask);
timevalue.it_interval.tv_sec = 1;
timevalue.it_interval.tv_usec = 0;
timevalue.it_value.tv_sec = 5;
timevalue.it_value.tv_usec = 0;
setitimer(ITIMER_REAL, &timevalue,NULL);
ret = sigaction(SIGALRM,&act,NULL);
if(ret < 0)
{
perror("sigaction");
return 0;
}
while(1)
{
}
return 0;
}
方式2
#include <signal.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/time.h>
void handle(int sig)
{
printf("Flappy bird\n");
}
int main(int argc,char *argv[])
{
struct itimerval timevalue;
timevalue.it_interval.tv_sec = 1;
timevalue.it_interval.tv_usec = 0;
timevalue.it_value.tv_sec = 5;
timevalue.it_value.tv_usec = 0;
setitimer(ITIMER_REAL, &timevalue,NULL);
signal(SIGALRM,handle);
while(1)
{
}
return 0;
}