当前位置: 首页 > 知识库问答 >
问题:

NA_real_和NaN之间的区别

公良同
2023-03-14

当我使用时。内部(inspect())NA_realNaN,它返回,

> .Internal(inspect(NA_real_))
@0x000001e79724d0e0 14 REALSXP g0c1 [REF(2)] (len=1, tl=0) nan
> .Internal(inspect(NaN))
@0x000001e797264a88 14 REALSXP g0c1 [REF(2)] (len=1, tl=0) nan

似乎他们唯一的区别是内存地址。

但是,当我强制将NA_real_uuNaN转换为字符时,它返回,

> as.character(c(NaN, NA_real_))
[1] "NaN" NA

我知道它应该返回上述结果,因为NaN不能是字符,它将被强制为“NaN”,但NA_real将被强制为NA_字符。但考虑到他们的直觉是一样的,R怎么能为他们返回不同的结果呢?

提前感谢您的任何建议!

共有2个答案

祁修诚
2023-03-14

NA是一个统计或数据完整性概念:缺失值的概念。例如,如果你的数据来自填写表单的人,一个错误的条目或丢失的条目将被视为NA

NaN是一个数字或计算概念:不是数字的东西。例如0/0是NAN,因为此计算的结果是未定义的(但请注意,1/0是Inf,或无穷大,类似地-1/0是-Inf)。

R在内部处理这些概念的方式不是你应该关心的。

梁丘佑运
2023-03-14

好首先,请记住,NA是一个R概念,在C中没有等价物。因此,必要时,NA需要在C中以不同的方式表示。事实上,。Internal(inspect())没有做出这种区分,但这并不意味着它不是在其他地方做出的。事实上,。内部(inspect())使用Rprintf打印值的内部双浮点表示形式。事实上,R NA在C浮点类型中被编码为NaN值。

其次,您注意到“它们唯一的区别是内存地址。”-那又怎么样?至少在概念上,不同的内存地址足以区分NA和NaN,不需要更多。

但事实上,R通过不同的途径区分这些值。这是可能的,因为IEEE 754双精度浮点格式有多种不同的NaN表示,而R为NAs保留了特定的表示:

static double R_ValueOfNA(void)
{
    /* The gcc shipping with Fedora 9 gets this wrong without
     * the volatile declaration. Thanks to Marc Schwartz. */
    volatile ieee_double x;
    x.word[hw] = 0x7ff00000;
    x.word[lw] = 1954;
    return x.value;
}

和:

/* is a value known to be a NaN also an R NA? */
int attribute_hidden R_NaN_is_R_NA(double x)
{
    ieee_double y;
    y.value = x;
    return (y.word[lw] == 1954);
}

int R_IsNA(double x)
{
    return isnan(x) && R_NaN_is_R_NA(x);
}

int R_IsNaN(double x)
{
    return isnan(x) && ! R_NaN_is_R_NA(x);
}

src/main/算术.c

 类似资料:
  • 问题内容: Java何时打印Infinity,什么时候打印NaN? 为什么将1.0 / 0.0无限大而将((1.0 / 0.0)-(1.0 / 0.0))NaN和0.0f / 0.0f设为NaN? 两者有什么区别? 问题答案: 因为Java遵循已知的数学事实。1.0 / 0.0是infinity,但其他形式都是不确定的形式,Java表示为(不是数字)。

  • 问题内容: 我错放了太多次了,我想我一直忘记,因为我不知道两者之间的区别,只是一个给了我我期望的价值,而另一个却没有。 为什么是这样? 问题答案: 是的简写形式(尽管请注意,该表达式只会被计算一次。) 是的,即指定一元的到。 例子:

  • 问题内容: 因此,我有一段简单的代码可以打印出整数1-10: 然后,如果仅在第3行上更改一个运算符,它将打印出无限数量的1整数(我知道为什么会这样做)。为什么在运行第二个程序时没有出现语法错误?如果赋值运算符后面跟着一个加法运算符,它不会调用语法错误吗? 问题答案: 与相同, 只是意味着。

  • 问题内容: 有人可以解释一下 和 我不知道“确切”的含义 问题答案: 在这个例子中,什么都没有。当您具有多个具有相似名称的路径时,该参数将起作用: 例如,假设我们有一个显示用户列表的组件。我们还有一个用于创建用户的组件。的网址应嵌套在下。因此,我们的设置可能如下所示: 现在,这里的问题是,当我们转到路由器时,将通过所有定义的路由,并返回它找到的第一个匹配项。因此,在这种情况下,它将首先找到路线,然

  • 问题内容: 我很好奇printStackTrace()和toString()之间的区别是什么。乍一看,他们 似乎 做的完全相同。 码: 问题答案: 不,有重要区别!使用toString,您只有异常的类型和错误消息。使用printStackTrace()可以获得异常的整个堆栈跟踪,这对于调试非常有帮助。 System.out.println(toString())的示例: printStackTra