理解attr(x,“class”)和class(x)之间的区别

问题描述 投票:10回答:1

Background

我正在看RomainFrançois提出的jazz包。 Romain使用以下语法定义函数is_bare_vector

is_bare_vector <- function(x) {
  is_vector(x) && !is.object(x) && is.null(attr(x, "class"))
}

Question

对于:x <- 1

attr(x, "class")
# NULL

鉴于:

class(x)
# [1] "numeric"

我想理解为什么这两个函数提供不同的答案? ?attr的帮助引用了?class

?attr

请注意,某些属性(即classcommentdimdimnamesnamesrow.namestsp)是专门处理的,并且对可以设置的值有限制。 (请注意,对于通过级别替换功能应为因子设置的级别,情况并非如此。)

?class

许多R对象都有一个class属性,一个字符向量给出了对象继承的类的名称。 (函数oldClassoldClass<-获取并设置属性,也可以直接完成。)

r class oop primitive
1个回答
8
投票

你只需要在help("class")中进一步阅读:

如果对象没有class属性,它有一个隐式类,特别是“matrix”,“array”,“function”或“numeric”或者typeof(x)的结果(类似于mode(x))

显然,如果class属性是class()NULL也将返回隐式类;让我们检查一下C source code - 我们看到如果class属性的长度为0,它将获得隐式类:

SEXP R_data_class(SEXP obj, Rboolean singleString)
{
    SEXP value, klass = getAttrib(obj, R_ClassSymbol);
    int n = length(klass);
    if(n == 1 || (n > 0 && !singleString))
    return(klass);
    if(n == 0) {
    SEXP dim = getAttrib(obj, R_DimSymbol);
    int nd = length(dim);
    if(nd > 0) {
        if(nd == 2)
        klass = mkChar("matrix");
        else
        klass = mkChar("array");
    }
    else {
      SEXPTYPE t = TYPEOF(obj);
      switch(t) {
      case CLOSXP: case SPECIALSXP: case BUILTINSXP:
        klass = mkChar("function");
        break;
      case REALSXP:
        klass = mkChar("numeric");
        break;
      case SYMSXP:
        klass = mkChar("name");
        break;
      case LANGSXP:
        klass = lang2str(obj, t);
        break;
      default:
        klass = type2str(t);
      }
    }
    }
    else
    klass = asChar(klass);
    PROTECT(klass);
    value = ScalarString(klass);
    UNPROTECT(1);
    return value;
}
© www.soinside.com 2019 - 2024. All rights reserved.