日期:2014-05-16  浏览次数:20656 次

读apue请教关于程序清单10-6的问题
本帖最后由 fxgcquhx 于 2013-03-15 11:04:14 编辑
最近读apue,在看代码即程序清单10-6(apue 中文版 第三版 254页)代码有疑问,请apue大牛帮助?
代码如下:
#include "apue.h"

unsigned int        sleep2(unsigned int);
static void         sig_int(int);

int
main(void)
{
    unsigned int        unslept;

    if (signal(SIGINT, sig_int) == SIG_ERR)
        err_sys("signal(SIGINT) error");
    unslept = sleep2(5);
    printf("sleep2 returned: %u\n", unslept);
    exit(0);
}

static void
sig_int(int signo)
{
    int            i, j;
    volatile int   k;

    /*
     * Tune these loops to run for more than 5 seconds
     * on whatever system this test program is run.
     */
    printf("\nsig_int starting\n");
    for (i = 0; i < 300000; i++)
        for (j = 0; j < 4000; j++)
            k += i * j;
    printf("sig_int finished\n");
}

问题1:书上说longjmp会提早终止该信号处理程序怎么理解?
      我的理解是longjmp跳转到sleep函数帧时,会将中断的那个信号处理程序即sig_int(...)函数帧抛弃掉,因此sig_int(...)信号处理函数不能继续运行,不知道是不是这样?
问题2:书上说“将整型变量k声明为volatitle,这样阻止了优化编译器丢弃循环语句“,这句话不知道怎么理解?
     我理解将k声明为volatitle可以保证k始终从内存中读数据,但是组织优化编译器丢弃循环语句就不明白了~~~
apue?c程序 signal 编译器 system

------解决方案--------------------
问题1差不多按照你的理解,loingjump这种非局部跳转会跳过栈上的若干栈,从而栈上原本的某些函数就会被抛弃掉了
问题2是这种双重循环在经过编绎器优化后,可能并非以双重循环的形式实现,比如可能以简单的一重循环来实现...当k声明为volatile 后, 表明k必须每次都需要更新,所以编绎器无法将双重循环优化掉,具体的还是要参见编绎原理和不同编绎器的实现..