zoukankan      html  css  js  c++  java
  • 信号

    信号的特质:由于信号是通过软件方法实现,其实现手段导致信号有很强的延时性。但对于用户来说,这个延迟时间非常短,不易察觉。

    每个进程收到的所有信号,都是由内核负责发送的,内核处理。

    产生信号:

      1. 按键产生,如:Ctrl+cCtrl+zCtrl+

      2. 系统调用产生,如:killraiseabort

      3. 软件条件产生,如:定时器alarm

      4. 硬件异常产生,如:非法访问内存(段错误)、除0(浮点数例外)、内存对齐出错(总线错误)

      5. 命令产生,如:kill命令

    递达:递送并且到达进程。

    未决:产生和递达之间的状态。主要由于阻塞(屏蔽)导致该状态。

    信号的处理方式: 

      1. 执行默认动作

      2. 忽略(丢弃) 

      3. 捕捉(调用户处理函数)

    PCB进程控制块中有未决信号集(set)、信号屏蔽字(阻塞信号集)

    阻塞信号集(信号屏蔽字) 将某些信号加入集合,对他们设置屏蔽,当屏蔽x信号后,再收到该信号,该信号的处理将推后(解除屏蔽后)

    未决信号集:

      1. 信号产生,未决信号集中描述该信号的位立刻翻转为1,表信号处于未决状态。当信号被处理对应位翻转回为0。这一时刻往往非常短暂。

      2. 信号产生后由于某些原因(主要是阻塞)不能抵达。这类信号的集合称之为未决信号集。在屏蔽解除前,信号一直处于未决状态。

    信号4要素

    与变量三要素类似的,每个信号也有其必备4要素,分别是:

    1. 编号 2. 名称 3. 事件 4. 默认处理动作

    默认动作:

      Term:终止进程

      Ign: 忽略信号 (默认即时对该种信号忽略操作)

      Core:终止进程,生成Core文件。(查验进程死亡原因, 用于gdb调试)

      Stop:停止(暂停)进程

      Cont:继续运行进程

    调试段错误:-g gdb a.out run (Core文件)

    这里特别强调了9) SIGKILL 19) SIGSTOP信号,不允许忽略和捕捉,只能执行默认动作。甚至不能将其设置为阻塞。

    终端按键产生信号

        Ctrl + c  2) SIGINT(终止/中断)  "INT" ----Interrupt

        Ctrl + z  20) SIGTSTP(暂停/停止)  "T" ----Terminal 终端。

        Ctrl +  3) SIGQUIT(退出)

    硬件异常产生信号

        0操作   → 8) SIGFPE (浮点数例外) "F" -----float 浮点数。

        非法访问内存  → 11) SIGSEGV (段错误)

        总线错误  → 7) SIGBUS

    kill函数/命令产生信号

    kill命令产生信号:kill -SIGKILL pid

    kill函数:给指定进程发送指定信号(不一定杀死)

      int kill(pid_t pid, int sig);  成功:0;失败:-1 (ID非法,信号非法,普通用户杀init进程等权级问题),设置errno

      sig:不推荐直接使用数字,应使用宏名,因为不同操作系统信号编号可能不同,但名称一致。

      pid > 0:  发送信号给指定的进程。

      pid = 0:  发送信号给 与调用kill函数进程属于同一进程组的所有进程。

      pid < 0:  |pid|发给对应进程组。

      pid = -1:发送给进程有权限发送的系统中所有进程。

    raiseabort函数

    raise 函数:给当前进程发送指定信号(自己给自己发) raise(signo) == kill(getpid(), signo);

         int raise(int sig); 成功:0,失败非0

    abort 函数:给自己发送异常终止信号 6) SIGABRT 信号,终止并产生core文件

         void abort(void); 该函数无返回

    alarm函数

    设置定时器(闹钟)。在指定seconds后,内核会给当前进程发送14SIGALRM信号。进程收到该信号,默认动作终止。

      每个进程都有且只有唯一个定时器。

      unsigned int alarm(unsigned int seconds); 返回0或剩余的秒数,无失败。

      常用:取消定时器alarm(0),返回旧闹钟余下秒数。

      定时,与进程状态无关(自然定时法)!就绪、运行、挂起(阻塞、暂停)、终止、僵尸...无论进程处于何种状态,alarm都计时。

      实际执行时间 = 系统时间 + 用户时间 + 等待时间

      使用time命令查看程序执行的时间。 程序运行的瓶颈在于IO,优化程序,首选优化IO

    setitimer函数

    设置定时器(闹钟)。 可代替alarm函数。精度微秒us,可以实现周期定时。

        int setitimer(int which, const struct itimerval *new_value, struct itimerval *old_value); 成功:0;失败:-1,设置errno

      参数:which:指定定时方式

      ① 自然定时:ITIMER_REAL 14SIGLARM   计算自然时间

      ② 虚拟空间计时(用户空间)ITIMER_VIRTUAL 26SIGVTALRM    只计算进程占用cpu的时间

      ③ 运行时计时(用户+内核)ITIMER_PROF 27SIGPROF  计算占用cpu及执行系统调用的时间

    信号集操作函数

    内核通过读取未决信号集来判断信号是否应被处理。信号屏蔽字mask可以影响未决信号集。而我们可以在应用程序中自定义set来改变mask。已达到屏蔽指定信号的目的。

    信号集(set)设定

      sigset_t  set; // typedef unsigned long sigset_t;

      int sigemptyset(sigset_t *set);     将某个信号集清0   成功:0;失败:-1

      int sigfillset(sigset_t *set);       将某个信号集置1    成功:0;失败:-1

      int sigaddset(sigset_t *set, int signum); 将某个信号加入信号集   成功:0;失败:-1

      int sigdelset(sigset_t *set, int signum); 将某个信号清出信号集    成功:0;失败:-1

      int sigismember(const sigset_t *set, int signum);判断某个信号是否在信号集中 返回值:在集合:1;不在:0;出错:-1  

      sigset_t类型的本质是位图。但不应该直接使用位操作,而应该使用上述函数,保证跨系统操作有效。

      对比认知select 函数。

    阻塞信号集只能通过我们自定义的信号集来设置

    sigprocmask函数

    用来屏蔽信号、解除屏蔽也使用该函数。其本质,读取或修改进程的信号屏蔽字(PCB)

      严格注意,屏蔽信号:只是将信号处理延后执行(延至解除屏蔽);而忽略表示将信号丢处理。

      int sigprocmask(int how, const sigset_t *set, sigset_t *oldset); 成功:0;失败:-1,设置errno

      参数

        set:传入参数,是一个位图,set中哪位置1,就表示当前进程屏蔽哪个信号。

        oldset:传出参数,保存旧的信号屏蔽集。

        how参数取值: 假设当前的信号屏蔽字为mask

      1. SIG_BLOCK: how设置为此值,set表示需要屏蔽的信号。相当于 mask = mask|set
      2. SIG _UNBLOCK: how设置为此,set表示需要解除屏蔽的信号。相当于 mask = mask & ~set

          3. SIG_SETMASK: how设置为此,set表示用于替代原始屏蔽及的新屏蔽集。相当于 mask = set若,调用sigprocmask解除了对当前若干个信号的阻塞,则在    sigprocmask返回前,至少将其中一个信号递达。

    sigpending函数

    读取当前进程的未决信号集

      int sigpending(sigset_t *set); set传出参数。   返回值:成功:0;失败:-1,设置errno

    sigaction函数

    修改信号处理动作(通常在Linux用其来注册一个信号的捕捉函数)

      int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);  成功:0;失败:-1,设置errno

      参数:

        act:传入参数,新的处理方式。

        oldact:传出参数,旧的处理方式

    struct sigaction结构体

        struct sigaction {

            void     (*sa_handler)(int);

            void     (*sa_sigaction)(int, siginfo_t *, void *);

            sigset_t   sa_mask;

            int       sa_flags;

            void     (*sa_restorer)(void);

        };

      sa_restorer:该元素是过时的,不应该使用,POSIX.1标准将不指定该元素。(弃用)

      sa_sigaction:当sa_flags被指定为SA_SIGINFO标志时,使用该信号处理程序。(很少使用)

    重点掌握:

      ① sa_handler:指定信号捕捉后的处理函数名(即注册函数)。也可赋值为SIG_IGN表忽略 或 SIG_DFL表执行默认动作

      ② sa_mask: 调用信号处理函数时,所要屏蔽的信号集合(信号屏蔽字)。注意:仅在处理函数被调用期间屏蔽生效,是临时性设置。

      ③ sa_flags:通常设置为0,表使用默认属性。

    信号捕捉特性

    1. 进程正常运行时,默认PCB中有一个信号屏蔽字,假定为☆,它决定了进程自动屏蔽哪些信号。当注册了某个信号捕捉函数,捕捉到该信号以后,要调用该函数。而该函数有可能执行很长时间,在这期间所屏蔽的信号不由☆来指定。而是用sa_mask来指定。调用完信号处理函数,再恢复为☆。
    2. XXX信号捕捉函数执行期间XXX信号自动被屏蔽

       3.阻塞的常规信号不支持排队,产生多次只记录一次。(后32个实时信号支持排队)

  • 相关阅读:
    MySQL for Linux错误: ERROR 1045 (28000): Access denied for user 'root'@'localhost' (using password: YES)
    Git SSH Key 生成步骤
    百度陆奇最新内部演讲:如何成为一个优秀的工程师?
    Apple使用Apache Mesos重建Siri后端服务
    剖析Elasticsearch集群系列第一篇 Elasticsearch的存储模型和读写操作
    ElasticSearch VS Solr
    聊聊基于Lucene的搜索引擎核心技术实践
    如何安全的存储用户密码?
    MySQL 性能管理及架构设计指南
    大牛是怎么思考设计MySQL优化方案
  • 原文地址:https://www.cnblogs.com/lr1402585172/p/10526346.html
Copyright © 2011-2022 走看看