-1

NAN来自的宏math.h是安静的 NAN:

ISO/IEC 9899:2011 (E)(强调):

当且仅当实现支持浮点类型的安静 NaN 时才定义。它扩展为表示安静 NaN的 float 类型的常量表达式。

安静的 NaN 通常不会导致引发 FP 异常。例子:

  1. ISO/IEC 9899:2011 (E)(强调):

5.2.4.2.2 浮动类型的特点<float.h>

3 一个安静的 NaN在几乎所有算术运算中传播,而不会引发浮点异常;当作为算术操作数出现时,信号 NaN 通常会引发浮点异常。

  1. IEEE 754-2008(已添加重点):

5.11 比较谓词的细节 明确考虑安静 NaN操作数可能性的程序可以使用表 5.3 中的无序安静谓词,它们不会发出这种无效操作异常的信号

然而:

  1. llrintf()既不是算术运算,也不是表 5.3 中的无序安静谓词。因此,5.2.4.2.2.35.11不适用。
  2. 7.12.9.5 The lrint and llrint functions(例如)没有说明如果输入是安静的 NaN 是否引发 FP 异常。

初步结论:由于一般做法“安静的 NaN 不会导致引发 FP 异常”,因此可以得出结论,如果输入是安静的 NaN,则lrintandllrint函数不应导致引发 FP 异常。

实践:

代码(t125.c):

#include <fenv.h>
#include <math.h>
#include <stdio.h>

#if _MSC_VER
#pragma fenv_access (on)
#else
#pragma STDC FENV_ACCESS ON
#endif

void show_fe_exceptions(void)
{
    printf("exceptions raised: ");
    if (fetestexcept(FE_DIVBYZERO))     printf(" FE_DIVBYZERO");
    if (fetestexcept(FE_INEXACT))       printf(" FE_INEXACT");
    if (fetestexcept(FE_INVALID))       printf(" FE_INVALID");
    if (fetestexcept(FE_OVERFLOW))      printf(" FE_OVERFLOW");
    if (fetestexcept(FE_UNDERFLOW))     printf(" FE_UNDERFLOW");
    if (fetestexcept(FE_ALL_EXCEPT)==0) printf(" none");
    printf("\n");
}

int main(void)
{
    long long ll;

    ll = llrintf(NAN);
    show_fe_exceptions();
    printf("ll %lld\n", ll);
    return 0;
}

调用:

$ cl t125.c /std:c11 /fp:strict && t125
exceptions raised:  FE_INEXACT FE_INVALID FE_OVERFLOW
ll 0

$ clang t125.c -std=c11 -ffp-model=strict -Wall -Wextra -pedantic && ./a.exe
t125.c:6:9: warning: unknown pragma ignored [-Wunknown-pragmas]
#pragma fenv_access (on)
        ^
1 warning generated.
exceptions raised:  FE_INEXACT FE_INVALID FE_OVERFLOW
ll -9223372036854775808

$ gcc t125.c -std=c11 -Wall -Wextra -pedantic && ./a.exe
t125.c:8: warning: ignoring ‘#pragma STDC FENV_ACCESS’ [-Wunknown-pragmas]
    8 | #pragma STDC FENV_ACCESS ON
      |
exceptions raised:  FE_INVALID
ll -9223372036854775808

llrintf问题:如果输入是安静的 NaN ,有人可以澄清围绕 C 标准库函数(例如)引发 FP 异常的情况/行为吗?

4

1 回答 1

0

OP 的印象是术语转换仅意味着隐式转换显式转换(强制转换操作)(请参阅 参考资料C11 6.3 Conversions)。这意味着 OP 的印象是 term function不是conversion

但是,C11 F.3 Operators and functions(强调)明确指出:

lrint和中的llrint 函数<math.h>提供 IEC 60559转换

因此,是的,C11 F.4 Floating to integer conversion(强调)回答了这个问题:

F.4 浮点数到整数的转换

1 ... 否则,如果浮点值是无穷大或NaN,或者如果浮点值的整数部分超出整数类型的范围,则引发''invalid''浮点异常并且结果值未指定. ...

于 2021-03-30T16:13:33.840 回答