`

linux c学习笔记----信号(sigaction,sigaddset,sigprocmask)

阅读更多

 



sigaction(查询或设置信号处理方式)
相关函数
signal,sigprocmask,sigpending,sigsuspend
表头文件
#include<signal.h>
定义函数
int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact);
函数说明

sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。
如参数结构sigaction定义如下


struct sigaction
{
void (*sa_handler) (int);
sigset_t sa_mask;
int sa_flags;
void (*sa_restorer) (void);
}


sa_handler此参数和signal()的参数handler相同,代表新的信号处理函数,其他意义请参考signal()。
sa_mask 用来设置在处理该信号时暂时将sa_mask 指定的信号搁置。
sa_restorer 此参数没有使用。
sa_flags 用来设置信号处理的其他相关操作,下列的数值可用。
OR 运算(|)组合
A_NOCLDSTOP : 如果参数signum为SIGCHLD,则当子进程暂停时并不会通知父进程
SA_ONESHOT/SA_RESETHAND:当调用新的信号处理函数前,将此信号处理方式改为系统预设的方式。
SA_RESTART:被信号中断的系统调用会自行重启
SA_NOMASK/SA_NODEFER:在处理此信号未结束前不理会此信号的再次到来。
如果参数oldact不是NULL指针,则原来的信号处理方式会由此结构sigaction 返回。

返回值
执行成功则返回0,如果有错误则返回-1。
错误代码
EINVAL 参数signum 不合法, 或是企图拦截SIGKILL/SIGSTOPSIGKILL信号
EFAULT 参数act,oldact指针地址无法存取。
EINTR 此调用被中断
范例
#include <stdio.h>
#include <signal.h>
#include <unistd.h>

void show_handler(int sig)
{
    printf("I got signal %d\n", sig);
    int i;
    for(i = 0; i < 5; i++) {
        printf("i = %d\n", i);
        sleep(1);
    }
}
 
int main(void)
{
    int i = 0;
    struct sigaction act, oldact;
    act.sa_handler = show_handler;
    sigaddset(&act.sa_mask, SIGQUIT); //见注(1)
    act.sa_flags = SA_RESETHAND | SA_NODEFER; //见注(2)
    //act.sa_flags = 0; //见注(3)

    sigaction(SIGINT, &act, &oldact);
    while(1) {
        sleep(1);
        printf("sleeping %d\n", i);
        i++;
    }
}
 

 

sigaddset(增加一个信号至信号集)

 

相关函数
sigemptyset,sigfillset,sigdelset,sigismember
表头文件
#include<signal.h>
定义函数
int sigaddset(sigset_t *set,int signum);
函数说明
sigaddset()用来将参数signum 代表的信号加入至参数set 信号集里。
返回值
执行成功则返回0,如果有错误则返回-1。
错误代码
EFAULT 参数set指针地址无法存取
EINVAL 参数signum非合法的信号编号

 

 

sigprocmask(查询或设置信号遮罩)

 

相关函数
signal,sigaction,sigpending,sigsuspend
表头文件
#include<signal.h>
定义函数
int sigprocmask(int how,const sigset_t *set,sigset_t * oldset);
函数说明
sigprocmask()可以用来改变目前的信号遮罩,其操作依参数how来决定
SIG_BLOCK 新的信号遮罩由目前的信号遮罩和参数set 指定的信号遮罩作联集
SIG_UNBLOCK 将目前的信号遮罩删除掉参数set指定的信号遮罩
SIG_SETMASK 将目前的信号遮罩设成参数set指定的信号遮罩。
如果参数oldset不是NULL指针,那么目前的信号遮罩会由此指针返回。

返回值
执行成功则返回0,如果有错误则返回-1。
错误代码
EFAULT 参数set,oldset指针地址无法存取。
EINTR 此调用被中断

实例:

 

#include <signal.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#define PROMPT "你想终止程序吗?"
char *prompt=PROMPT;
void ctrl_c_op(int signo)
{
write(STDERR_FILENO,prompt,strlen(prompt));
}
int main()
{
struct sigaction act;
act.sa_handler=ctrl_c_op;
sigemptyset(&act.sa_mask);
act.sa_flags=0;
if(sigaction(SIGINT,&act,NULL)<0)
{
fprintf(stderr,"Install Signal Action Error:%s\n\a",strerror(errno));
exit(1);
}
while(1);
}

#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <fcntl.h>
#include <signal.h>
#include <string.h>
#include <pwd.h>
#include <sys/types.h>
#include <sys/stat.h>
/* Linux 的默任个人的邮箱地址是 /var/spool/mail/ */
#define MAIL_DIR "/var/spool/mail/"
/* 睡眠10 秒钟 */
#define SLEEP_TIME 10
#define MAX_FILENAME 255
unsigned char notifyflag=1;

long get_file_size(const char *filename){
	struct stat buf;
	if(stat(filename,&buf)==-1){
		if(errno==ENOENT)return 0;
			else return -1;
	}
	return (long)buf.st_size;
}

void send_mail_notify(void){
	fprintf(stderr,"New mail has arrived\007\n");
}

void turn_on_notify(int signo){
	notifyflag=1;
}

void turn_off_notify(int signo){
	notifyflag=0;
}
int check_mail(const char *filename){
	long old_mail_size,new_mail_size;
	sigset_t blockset,emptyset;
	sigemptyset(&blockset);
	sigemptyset(&emptyset);
	sigaddset(&blockset,SIGUSR1);
	sigaddset(&blockset,SIGUSR2);
	old_mail_size=get_file_size(filename);
	if(old_mail_size<0)return 1;
	if(old_mail_size>0) send_mail_notify();
	sleep(SLEEP_TIME);
	while(1){
		if(sigprocmask(SIG_BLOCK,&blockset,NULL)<0) return 1;
		while(notifyflag==0)sigsuspend(&emptyset);
		if(sigprocmask(SIG_SETMASK,&emptyset,NULL)<0) return 1;
		new_mail_size=get_file_size(filename);
		if(new_mail_size>old_mail_size)send_mail_notify;
		old_mail_size=new_mail_size;
		sleep(SLEEP_TIME);
	}
}

int main(void){
char mailfile[MAX_FILENAME];
struct sigaction newact;
struct passwd *pw;
if((pw=getpwuid(getuid()))==NULL){
	fprintf(stderr,"Get Login Name Error:%s\n\a",strerror(errno));
	exit(1);
}
strcpy(mailfile,MAIL_DIR);
strcat(mailfile,pw->pw_name);
newact.sa_handler=turn_on_notify;
newact.sa_flags=0;
sigemptyset(&newact.sa_mask);
sigaddset(&newact.sa_mask,SIGUSR1);
sigaddset(&newact.sa_mask,SIGUSR2);
if(sigaction(SIGUSR1,&newact,NULL)<0)
	fprintf(stderr,"Turn On Error:%s\n\a",strerror(errno));
newact.sa_handler=turn_off_notify;
if(sigaction(SIGUSR1,&newact,NULL)<0)
	fprintf(stderr,"Turn Off Error:%s\n\a",strerror(errno));
check_mail(mailfile);
exit(0);
}
这个程序会在也可以检查用户的邮件.不过提供了一个开关,如果用
户不想程序提示有新的邮件到来,可以向程序发送SIGUSR2 信号,如果想程序提供提示可以
发送SIGUSR1 信号.
 
分享到:
评论

相关推荐

    Linux下C语言编程--时间概念

    在Linux环境下进行C语言编程时,时间概念是一个重要的部分,涉及到如何获取、处理和测量时间。以下是对这些知识点的详细说明: 1. **时间表示**: 在C语言中,时间通常通过`time_t`类型表示,它是一个整型变量,...

    01--Linux系统编程-信号.docx

    Linux系统编程中的信号机制是操作系统提供的一种异步通信方式,它允许进程间或者操作系统与进程之间传递简短的通知。信号的概念源于现实生活中的一些信号行为,它们具有意图简单、信息量小且满足特定触发条件的特点...

    linux-c-api-ref.zip_linux api_linux-c-api-ref

    4. **信号处理**:信号是进程间通信的一种方式,`signal()`或`sigaction()`用于注册信号处理函数,`raise()`发送信号给当前进程,`kill()`向指定进程发送信号。 5. **内存管理**:`malloc()`和`free()`负责动态内存...

    Linux下C语言应用编程--随书源代码

    5. **信号处理**:信号是Linux系统中进程间通信的一种方式,通过signal、raise、sigaction等函数可以捕获和处理信号,实现特定的行为。 6. **系统调用**:Linux提供了丰富的系统调用,如open、close、read、write等...

    Linux系统下C语言编程--进程的创建

    此外,C语言编程中还有其他与进程控制相关的系统调用,例如`exec()`系列函数用于替换进程的执行映像,`wait()`和`waitpid()`用于父进程等待子进程结束并回收资源,以及`signal()`和`sigaction()`处理进程中的信号。...

    Linux下的C-C++编程

    10. **信号处理**:信号是Linux中进程间通信的一种方式,通过`signal()`和`sigaction()`等函数来捕获和处理信号。 11. **内存管理**:动态内存分配(`malloc()`、`calloc()`、`realloc()`和`free()`)是C/C++编程中...

    C例子:IPC-sigaction

    该程序是我写的博客“一起talk C栗子吧(v第八十六回:C语言实例--使用信号进行进程间通信三)”的配套程序,共享给大家使用

    linux-api-2.6.22.rar_linux_linux api_linux-2.6.22.6

    7. **定时器和信号**:`setitimer()`和`getitimer()`用于设置和获取定时器,而`signal()`和`sigaction()`处理信号。这些功能在处理异步事件和超时处理中起着核心作用。 8. **内核模块**:Linux 2.6.22.6支持可加载...

    信号捕捉函数sigaction

    ### 信号捕捉函数sigaction详解 #### 一、引言 在计算机编程中,尤其是在Linux环境下,信号(Signal)作为一种重要的通信机制被广泛应用于进程间通信。为了更好地控制信号的行为,POSIX标准引入了`sigaction`函数...

    Linux进程间通信-信号通信信号发送实例.pdf

    `sigaction`函数用于设置信号处理函数和信号行为,`sigqueue`则用于携带自定义数据发送信号。程序会持续等待信号并每两秒发送一次,附加的数据是字符串"other info"。 在实际应用中,信号通信可以结合其他IPC机制,...

    linux-c-develop.rar

    在Linux环境下进行C语言开发是一项基础且重要的技能,尤其对于那些致力于系统级编程或嵌入式领域的开发者。"linux-c-develop.rar"这个压缩包显然包含了关于Linux操作系统下C程序开发的入门指南,对于初学者来说是一...

    linux信号专题笔记.docx

    ### Linux信号专题笔记 #### Linux信号基本概念 Linux信号是一种软件中断机制,它允许系统对特定事件做出反应。这些事件可以是程序错误、外部请求或者是系统级别的改变。当发生这样的事件时,系统会向进程发送一个...

    Linux系统编程学习笔记

    ### Linux系统编程学习笔记 #### 一、IO **1.1 标准I/O (stdio)** - **fopen/fclose**: `fopen` 用于打开或创建一个文件,并返回一个指向该文件的 `FILE *` 类型的指针。`fclose` 用于关闭一个已经打开的文件。...

    LINUX编程白皮书001-03

    `signal()`和`sigaction()`函数用于安装信号处理器,`raise()`和`kill()`用于发送信号。理解信号机制对于编写响应系统事件的程序至关重要。 网络编程在Linux环境中也是重要的一环。`socket()`函数创建套接字,`bind...

    linux-api-2.6.22.rar_linux-api-2.6.22_linux-api-2.6.22.chm

    4. **信号处理**:signal()、raise()、sigaction(),用于处理进程间的通信和异常情况。 5. **网络编程**:socket()、bind()、connect()、listen()、accept()等,用于创建套接字、绑定地址、连接服务器、监听连接请求...

    Linux进程间通信-信号通信实例.pdf

    在学习和实践中,了解和掌握信号通信是理解Linux多进程交互的关键步骤。参考文献如《基于项目驱动的嵌入式Linux应用设计开发》等资源可以帮助深化这一领域的知识。在实际编程时,应根据具体需求选择合适的信号处理...

    Linux-C-Reference-Manual.rar_C Reference Manual_c a reference ma

    《Linux C参考手册》是为C语言程序员在Linux环境下编程所准备的重要参考资料。这份手册详尽地涵盖了Linux系统中C语言编程的各种函数、标准库、系统调用以及相关的编程概念。通过对这份手册的学习,开发者可以深入...

    Linux-c课件

    "Linux C课件"涵盖了这一关键领域的知识,帮助学习者掌握在Linux环境下使用C语言进行程序设计的基础与进阶技巧。 1. **Linux环境介绍** - Linux是一种自由开放的操作系统,基于Unix,具有高度可定制性和稳定性。 ...

Global site tag (gtag.js) - Google Analytics