r - 如何在不知道 Rcpp 类型的情况下处理向量

标签 r rcpp

我想在 Rcpp 中复制以下 R 函数:

fR = function(x) x[1:2]

fR(c(1,2,3))
#[1] 1 2
fR(c('a','b','c'))
#[1] "a" "b"

我可以为固定输出类型执行此操作,如下所示:
library(inline)
library(Rcpp)

fint = cxxfunction(signature(x = "SEXP"), '
          List xin(x);
          IntegerVector xout;

          for (int i = 0; i < 2; ++i) xout.push_back(xin[i]);

          return xout;', plugin = "Rcpp")

但这仅适用于整数,如果我尝试替换 xout输入 List (或 GenericVector ,它们是相同的) - 它适用于任何输入类型,但我得到一个 list而不是向量。

什么是正确的 Rcpp这样做的方式?

最佳答案

不要使用 push_backRcpp类型。当前实现 Rcpp 向量的方式需要每次复制所有数据。这是一项非常昂贵的操作。

我们有 RCPP_RETURN_VECTOR对于调度,这要求您编写一个模板函数,将 Vector 作为输入。

#include <Rcpp.h>
using namespace Rcpp ;

template <int RTYPE>
Vector<RTYPE> first_two_impl( Vector<RTYPE> xin){
    Vector<RTYPE> xout(2) ;
    for( int i=0; i<2; i++ ){
        xout[i] = xin[i] ;    
    }
    return xout ;
}

// [[Rcpp::export]]
SEXP first_two( SEXP xin ){
  RCPP_RETURN_VECTOR(first_two_impl, xin) ;
}

/*** R
    first_two( 1:3 )
    first_two( letters )
*/

刚刚sourceCpp这个文件,这也将运行调用这两个函数的 R 代码。实际上,模板可以更简单,这也可以:
template <typename T>
T first_two_impl( T xin){
    T xout(2) ;
    for( int i=0; i<2; i++ ){
        xout[i] = xin[i] ;    
    }
    return xout ;
}

模板参数T只需要:
  • 一个接受 int 的构造函数
  • operator[](int)

  • 或者,这可能是 dplyr 的工作矢量访客。
    #include <dplyr.h>
    // [[Rcpp::depends(dplyr,BH)]]
    
    using namespace dplyr ;
    using namespace Rcpp ;
    
    // [[Rcpp::export]]
    SEXP first_two( SEXP data ){
        VectorVisitor* v = visitor(data) ;
        IntegerVector idx = seq( 0, 1 ) ;
        Shield<SEXP> out( v->subset(idx) ) ;
        delete v ;
        return out ;
    }
    

    访问者让您可以在向量上执行一组操作,而不管其保存的数据类型如何。
    > first_two(letters)
    [1] "a" "b"
    
    > first_two(1:10)
    [1] 1 2
    
    > first_two(rnorm(10))
    [1] 0.4647190 0.9790888
    

    关于r - 如何在不知道 Rcpp 类型的情况下处理向量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19823915/

    相关文章:

    R - ggplot 上的背景颜色沿对角线分割?

    r - 禁用 y 轴刻度和标签

    rcpp编译错误

    r - RcppEigen中的有效加权协方差

    Rcpp 和 R : pass by reference

    c++ - 在不知道实际内部数据类型的情况下返回给定 SEXP 的子集

    c++ - 分配大 vector 的问题

    在 R 中重复表达式几次

    R,对具有重复条目的连续行的数据帧进行操作

    r - 使用 rvest 选择特定样式