如何将指向函数的指针从C代码传递给R(使用外部R)并在从R ??调用该函数之后
类似的东西:
C:
typedef void (* FunctionPtr)();
SEXP ans;
PROTECT(ans = /* ?some code? */);
R_tryEval(ans, R_GlobalEnv, NULL);
UNPROTECT(1);
R:
callback_function()
修改
@Romain Francois的帖子非常有用。
myapp代码:
namespace
{
void callback()
{
std::cout << "callback\n" << std::flush;
}
}
class Worker
{
public:
/*...*/
void initialize(argc, argv)
{
Rf_initEmbeddedR(argc, argv);
SEXP ans, val;
typedef void (* FunctionPtr)();
PROTECT(ans = Rf_lang2(Rf_install("source"), Rf_mkString("script.R")));
R_tryEval(ans, R_GlobalEnv, NULL);
UNPROTECT(1);
PROTECT(val = Rf_ScalarInteger((int)(&callback)));
/* pass the address of the pointer to a function */
PROTECT(ans = Rf_lang2(Rf_install("setCallback"), val));
R_tryEval(ans, R_GlobalEnv, NULL);
UNPROTECT(2);
}
void uninitialize()
{
Rf_endEmbeddedR(0);
}
};
R和Rcpp
的 script.R
###################
sourceCpp("utils.cpp")
###################
callback <- function()
{
callCallback()
}
utils.cpp
#include <Rcpp.h>
using namespace Rcpp;
typedef void (* Callback)();
static Callback spCallback = 0;
// [[Rcpp::export]]
void callCallback()
{
if (spCallback) {
spCallback();
} else {
Rprintf("ERROR: callback is not set");
}
}
// [[Rcpp::export]]
void setCallback(const int address)
{
spPlaceOrder = (Callback)address;
}
答案 0 :(得分:6)
您正在寻找外部指针。他们会让你封装指向任意数据结构的指针。 数据这个词在这里很重要,而且函数指针是一个不同的野兽。如果你想使用C++
和Rcpp
,我建议创建一个封装函数指针的小类:
typedef void (* FunctionPtr)();
class FunctionPointer {
FunctionPtr ptr;
public:
FunctionPointer( FunctionPtr ptr_) : ptr(ptr_){}
} ;
然后创建XPtr<FunctionPointer>
:
#include <Rcpp.h>
using namespace Rcpp ;
// your callback function
void callback(){
Rprintf( "hello from callback\n" ) ;
}
// The function that creates an external pointer to your
// callback
// [[Rcpp::export]]
XPtr<FunctionPointer> create_ptr(){
return XPtr<FunctionPointer>( new FunctionPointer(callback) );
}
// The function that invokes the callback
// [[Rcpp::export]]
void invokeCallback( XPtr<FunctionPointer> callback){
callback->ptr() ;
}
在R端,您可以使用词法作用域将外部指针包装到R函数中:
callback <- local( {
ptr <- create_ptr()
function(){
invokeCallback( ptr )
invisible(NULL)
}
} )
callback()
答案 1 :(得分:4)
另一种方法是使用InternalFunction
中的Rcpp
类:
#include <Rcpp.h>
using namespace Rcpp ;
void callback(){
Rprintf( "hello from calback\n" ) ;
}
// [[Rcpp::export]]
InternalFunction get_callback(){
return InternalFunction(callback) ;
}
实现类似于我在其他答案中描述的内容。 R侧的事情由Rcpp
:
callback <- get_callback()
callback()
答案 2 :(得分:0)
首先加载lib:
dyn.load("yourDLL.dll")
然后使用:
调用已加载到R的已编译代码的函数。
请参阅The R FFI
编辑:
首先编译你的C代码。在Windows上如下,但在其他操作系统上类似:
gcc mylib.c -shared -o mylib.dll
然后在R:
dyn.load("mylib.dll")
tmp1 <- as.raw(rep(0,4)) # 4 bytes to be sent to the function's argument
tmp2<-.C("yourFunctionName", tmp1) # tmp2 now has the data returned by the func
请注意,在c ++(而不是c)中,函数名称必须是mangled name