我正在看RomainFrançois提出的jazz包。 Romain使用以下语法定义函数is_bare_vector
:
is_bare_vector <- function(x) {
is_vector(x) && !is.object(x) && is.null(attr(x, "class"))
}
对于:x <- 1
:
attr(x, "class")
# NULL
鉴于:
class(x)
# [1] "numeric"
我想理解为什么这两个函数提供不同的答案? ?attr
的帮助引用了?class
?attr
请注意,某些属性(即
class
,comment
,dim
,dimnames
,names
,row.names
和tsp
)是专门处理的,并且对可以设置的值有限制。 (请注意,对于通过级别替换功能应为因子设置的级别,情况并非如此。)
?class
许多R对象都有一个class属性,一个字符向量给出了对象继承的类的名称。 (函数
oldClass
和oldClass<-
获取并设置属性,也可以直接完成。)
你只需要在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;
}