gpt4 book ai didi

c++ - 如何获得 SEXP 值的 SEXPTYPE?

转载 作者:行者123 更新时间:2023-11-30 01:17:09 31 4
gpt4 key购买 nike

假设我有一个函数将 SEXP 类型作为参数:

SEXP myFun(SEXP param)

如何找出这个参数的类型?查看我的调试器中的 SEXP 类型,我可以看到我可以调用 param->sxpinfo.type 来获得 SEXPTYPE 的数字表示>。快速浏览一下,它们似乎与这些匹配:

no    SEXPTYPE    Description 
0 NILSXP NULL
1 SYMSXP symbols
2 LISTSXP pairlists
3 CLOSXP closures
4 ENVSXP environments
5 PROMSXP promises
6 LANGSXP language objects
7 SPECIALSXP special functions
8 BUILTINSXP builtin functions
9 CHARSXP internal character strings
10 LGLSXP logical vectors
13 INTSXP integer vectors
14 REALSXP numeric vectors
15 CPLXSXP complex vectors
16 STRSXP character vectors
17 DOTSXP dot-dot-dot object
18 ANYSXP make “any” args work
19 VECSXP list (generic vector)
20 EXPRSXP expression vector
21 BCODESXP byte code
22 EXTPTRSXP external pointer
23 WEAKREFSXP weak reference
24 RAWSXP raw vector
25 S4SXP S4 classes not of simple type

(来源:http://www.biosino.org/R/R-doc/R-ints/SEXPTYPEs.html#SEXPTYPEs)

但这看起来很老套。检查 SEXP 变量类型的正确方法是什么?

最佳答案

对于 R API,TYPEOF 宏用于获取运行时类型。我们可以从 Rinternals.h 中看到一些相关位(有趣的是,不是编码为 enum,而是作为一系列宏定义;大概是为了在一些非常糟糕的平台上向后兼容一些非常糟糕的编译器......)

typedef unsigned int SEXPTYPE;

#define NILSXP 0 /* nil = NULL */
#define SYMSXP 1 /* symbols */
#define LISTSXP 2 /* lists of dotted pairs */
#define CLOSXP 3 /* closures */
#define ENVSXP 4 /* environments */
#define PROMSXP 5 /* promises: [un]evaluated closure arguments */
#define LANGSXP 6 /* language constructs (special lists) */
#define SPECIALSXP 7 /* special forms */
#define BUILTINSXP 8 /* builtin non-special forms */
#define CHARSXP 9 /* "scalar" string type (internal only)*/
#define LGLSXP 10 /* logical vectors */
/* 11 and 12 were factors and ordered factors in the 1990s */
#define INTSXP 13 /* integer vectors */
#define REALSXP 14 /* real variables */
#define CPLXSXP 15 /* complex variables */
#define STRSXP 16 /* string vectors */
#define DOTSXP 17 /* dot-dot-dot object */
#define ANYSXP 18 /* make "any" args work.
Used in specifying types for symbol
registration to mean anything is okay */
#define VECSXP 19 /* generic vectors */
#define EXPRSXP 20 /* expressions vectors */
#define BCODESXP 21 /* byte code */
#define EXTPTRSXP 22 /* external pointer */
#define WEAKREFSXP 23 /* weak reference */
#define RAWSXP 24 /* raw bytes */
#define S4SXP 25 /* S4, non-vector */

/* used for detecting PROTECT issues in memory.c */
#define NEWSXP 30 /* fresh node creaed in new page */
#define FREESXP 31 /* node released by GC */

#define FUNSXP 99 /* Closure or Builtin or Special */

如果定义了 USE_RINTERNALS,我们可以看到 R 使用以下命令查询 SEXPTYPE:

#define TYPEOF(x)   ((x)->sxpinfo.type)

这正是你所提议的:)但是在大多数情况下(即——除非你知道你在做什么),你不应该使用那个#define,所以定义来自memory.c :

int (TYPEOF)(SEXP x) { return TYPEOF(CHK(x)); }

...它只是调用 TYPEOF 宏,但使用 CHK 来确保它正在查看的 SEXP 没有已经不 protected 。

浏览 R 源代码(一定要看看 R.hRinternals.h)对于更好地了解 Rinternals.h 中实际暴露的内容很有用R API 及其使用方式。

也就是说,R API 有点丑陋,所以我们确实推荐使用 Rcpp,它提供了许多包装 SEXP 的不错的类s 但提供编译时类型和一系列有用的函数来使用/操作它们。参见@eddelbuettel 的 Rcpp page进行介绍,以及 the Rcpp Gallery例如 Rcpp 使用。

关于c++ - 如何获得 SEXP 值的 SEXPTYPE?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25172419/

31 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com