15

我对 Rcpp 模块有以下问题:假设我在 Rcpp 模块中有两个类

class A {
  public:
    int x;
};

class B
  public:
    A get_an_a(){
      A an_a();
      an_a.x=3;
      return an_a;
    }
};

RCPP_MODULE(mod){
  using namespace Rcpp ;
  class_<A>("A")
   .constructor()
   .property("x",&A::get_x)
  ;
  class_<B>("B)
   .constructor()
   .method("get_an_A",&get_an_a)
  ;
}

.

现在编译失败,因为它不知道如何处理 A 的返回类型。

我想我可以用 Rcpp::Xptr 做一些事情,但是,我无法将它连接到 Rcpp 为 A 类生成的 S4 结构。我实际上从 R 中的方法获得了一个外部指针对象。

是否有可能从第二类的方法中获取正确包装的对象?

谢谢,托马斯

[编辑]

根据 Dirk 的回答,我构建了一个可以创建包装的 S4 对象的包装器:

template <> SEXP wrap(const A &obj) { // insprired from "make_new_object" from Rcpp/Module.h
  Rcpp::XPtr<A> xp( new A(obj), true ) ; // copy and mark as finalizable
  Function maker=Environment::Rcpp_namespace()[ "cpp_object_maker"];
  return maker ( typeid(A).name() , xp );
}

不过,我不知道如何将对象作为方法/函数的参数返回。以下不起作用:

template <> A* as( SEXP obj){
  Rcpp::List l(obj);
  Rcpp::XPtr<A> xp( (SEXP) l[".pointer"] );
  return (A*) xp;
}

那么如何从参数列表中作为 SEXP 提​​供的 S4 对象中获取指向 C++ 对象的外部指针呢?

4

3 回答 3

13

此功能已添加到 Rcpp 0.10.0

您的代码无法编译还有其他原因。下面的代码对我有用:

class A {
  public:
    A(int x_) : x(x_){}

    int x;
};

class B {
  public:
    A get_an_a(){
      A an_a(3);
      return an_a;
    }
};
RCPP_EXPOSED_CLASS(A)
RCPP_EXPOSED_CLASS(B)

RCPP_MODULE(stackmod){
  using namespace Rcpp ;
  class_<A>("A")
   .constructor<int>()
   .field("x",&A::x)
  ;
  class_<B>("B")
   .constructor()
   .method("get_an_A",&B::get_an_a)
  ;
}

付出的代价是调用RCPP_EXPOSED_CLASS所有类的宏:

  • 您的模块之一正在暴露
  • 您想作为公开方法的结果或作为参数使用

有了这个,我得到:

> b <- new( B )
> b$get_an_A( )
C++ object <0x10330a9d0> of class 'A' <0x1006f46e0>
> b$get_an_A( )$x
[1] 3
于 2012-10-25T08:22:16.867 回答
3

如果你为它写一个包装器是可能的。那些不会像天上掉下来的吗哪,你需要通过提供它来帮助编译器,然后它就会被选中。

来自 RcppBDT 的简单示例:

template <> SEXP wrap(const boost::gregorian::date &d) {
    // convert to y/m/d struct
    boost::gregorian::date::ymd_type ymd = d.year_month_day();     
    return Rcpp::wrap(Rcpp::Date( ymd.year, ymd.month, ymd.day ));
}

在这里,Boost 中的一个类被转换为一个 Rcpp 类(Date),然后返回(即使在那里我们也需要一个显式包装)。对于更复杂的类,您需要更复杂的转换器。

编辑:当然,您需要记住,任何可从 R 调用的东西仍然需要SEXP foo(SEXP a, SEXP b, ...)符合.Call().

于 2012-09-13T12:07:11.427 回答
1

好的,我想我明白了。但没有任何保证。根据 Dirk 的回答,我构建了一个可以创建包装的 S4 对象的包装器:

template <> SEXP wrap(const A &obj) { // insprired from "make_new_object" from Rcpp/Module.h
  Rcpp::XPtr<A> xp( new A(obj), true ) ; // copy and mark as finalizable
  Function maker=Environment::Rcpp_namespace()[ "cpp_object_maker"];
  return maker ( typeid(A).name() , xp );
}

如果一个方法期望一个对象作为参数(这里是一个指向该对象的指针),那么下面的“as”包装器可能有用:

template <> A* as( SEXP obj){
  Rcpp::Environment e(obj);
  Rcpp::XPtr<A> xp( (SEXP) e.get(".pointer") );
  return (A*) xp;
}

您当然也可以返回 *xp ,它允许将非指针作为参数,但这又会复制对象。

于 2012-09-17T11:21:27.863 回答