编写可重入函数要注意的地方

    技术2022-05-11  106

    概念:

    什么是可重入函数:函数被多个进程调用。

    应注意的地方

    1:编写可重入函数时,应注意局部变量的使用(如编写C/C++语言的可重入函数时,应使用auto即缺省态局部变量或寄存器变量)。

    说明:编写C/C++语言的可重入函数时,不应使用static局部变量,否则必须经过特殊处理,才能使函数具有可重入性。

    2:编写可重入函数时,若使用全局变量,则应通过关中断、信号量(即PV操作)等手段对其加以保护。

    说明:若对所使用的全局变量不加以保护,则此函数就不具有可重入性,即当多个进程调用此函数时,很有可能使有关全局变量变为不可知状态。

    示例:假设Examint型全局变量,函数Squre_Exam返回Exam平方值。那么如下函数不具有可重入性。

    unsigned int example( int para )

    {

        unsigned int temp;

        Exam = para; // **

     temp = Square_Exam( );

     return temp;

    }

    此函数若被多个进程调用的话,其结果可能是未知的,因为当(**)语句刚执行完后,另外一个使用本函数的进程可能正好被激活,那么当新激活的进程执行到此函数时,将使Exam赋与另一个不同的para值,所以当控制重新回到“temp = Square_Exam( )”后,计算出的temp很可能不是预想中的结果。此函数应如下改进。

    unsigned int example( int para )

    {

        unsigned int temp;

     

        [申请信号量操作]          // 若申请不到“信号量”,说明另外的进程正处于

        Exam = para;            // Exam赋值并计算其平方过程中(即正在使用此

        temp = Square_Exam( );  // 信号),本进程必须等待其释放信号后,才可继

        [释放信号量操作]          // 续执行。若申请到信号,则可继续执行,但其

                                // 它进程必须等待本进程释放信号量后,才能再使

                                // 用本信号。

        return temp;

    }

    2:编写可重入函数时,若使用全局变量,则应通过关中断、信号量(即PV操作)等手段对其加以保护。

    说明:若对所使用的全局变量不加以保护,则此函数就不具有可重入性,即当多个进程调用此函数时,很有可能使有关全局变量变为不可知状态。

    示例:假设Examint型全局变量,函数Squre_Exam返回Exam平方值。那么如下函数不具有可重入性。

    unsigned int example( int para )

    {

        unsigned int temp;

        Exam = para; // **

     temp = Square_Exam( );

     return temp;

    }

    此函数若被多个进程调用的话,其结果可能是未知的,因为当(**)语句刚执行完后,另外一个使用本函数的进程可能正好被激活,那么当新激活的进程执行到此函数时,将使Exam赋与另一个不同的para值,所以当控制重新回到“temp = Square_Exam( )”后,计算出的temp很可能不是预想中的结果。此函数应如下改进。

    unsigned int example( int para )

    {

        unsigned int temp;

     

        [申请信号量操作]          // 若申请不到“信号量”,说明另外的进程正处于

        Exam = para;            // Exam赋值并计算其平方过程中(即正在使用此

        temp = Square_Exam( );  // 信号),本进程必须等待其释放信号后,才可继

        [释放信号量操作]          // 续执行。若申请到信号,则可继续执行,但其

                                // 它进程必须等待本进程释放信号量后,才能再使

                                // 用本信号。

        return temp;

    }


    最新回复(0)