10

系统规格:

  • 操作系统 - Mac OS X 10.6.8(雪豹)
  • g++ - Macports gcc 4.8.1_2+通用
  • R - 2.15.3
  • Rcpp - 0.10.3

当我尝试编译在 R 中(通过 Rcpp)使用 C++11 的函数时,我不断收到错误消息——由于某种原因,g++ 无法识别-std=c++11选项。

此示例取自 Rcpp 帮助文件(它不包含任何特定于 C++11 的内容,但可以显示我的问题所在)。如果我尝试运行:

require( Rcpp )
Sys.setenv( "PKG_CXXFLAGS"="-std=c++11" )
cppFunction(plugins=c("cpp11"), '
int useCpp11() {
    int x = 10;
    return x;
}')

我得到:

cc1plus: error: unrecognized command line option "-std=c++11"
make: *** [file61239328ae6.o] Error 1
g++ -arch x86_64 -I/Library/Frameworks/R.framework/Resources/include -I/Library/Frameworks/R.framework/Resources/include/x86_64 -DNDEBUG  -I/usr/local/include  -I"/Library/Frameworks/R.framework/Versions/2.15/Resources/library/Rcpp/include"   -std=c++11  -fPIC  -g -O2  -c file61239328ae6.cpp -o file61239328ae6.o 
Error in sourceCpp(code = code, env = env, rebuild = rebuild, showOutput = showOutput,  : 
  Error 1 occurred building shared library.

同时,我可以直接从 bash 编译这个函数——如果这个代码在useCpp11.cpp文件中,那么它运行没有任何抱怨:

g++ useCpp11.cpp -std=c++11

当然,我做错了什么,但我无法弄清楚它是什么。gcc 4.8 被设置为 bash 中的默认编译器,Rcpp 过去一直在正常工作。我怀疑我没有告诉 R 使用哪个版本的 g++ - 可能是这样吗?

4

3 回答 3

12

Kevin Ushley 是绝对正确的——确保使用正确的编译器的最简单方法是通过Makevars文件。就我而言,我补充说:

CXX = g++-4.8.1
PKG_CXXFLAGS = -std=c++11

这就是我所缺少的——后来一切都奏效了。如果您正在编译您的包,这将有效。

于 2013-08-27T05:19:46.077 回答
7

快速的:

  • 落后于发布版本 0.10.4 的 Rcpp

  • 您使用的版本(0.10.3)确实有 C++11 的插件

  • Rcpp Gallery 上有一篇完整的文章详细说明了这一点。

因此,请允许我引用Rcpp Gallery 上的 C++11 文章

R> library(Rcpp)
R> sourceCpp("/tmp/cpp11.cpp")
R> useAuto()
[1] 42
R> 

其中代码/tmp/cpp11.cpp如下:

#include <Rcpp.h>

// Enable C++11 via this plugin (Rcpp 0.10.3 or later)
// [[Rcpp::plugins("cpp11")]]

// [[Rcpp::export]]
int useAuto() {
    auto val = 42;      // val will be of type int
    return val;
}

如果这对您不起作用,那么您的系统设置不正确。换句话说,这不是Rcpp标签的问题——而是“如何设置路径来调用我认为应该调用的 g++ 版本”。

于 2013-08-27T00:45:37.363 回答
2

如果您不想修复路径,您可以PKG_CXXFLAGS在编译之前设置环境变量。

     export PKG_CXXFLAGS='`Rscript -e "Rcpp:::CxxFlags()"` -std=c++11'

例如:

R中

#generate the bare Rcpp package as usual
library(Rcpp)
Rcpp.package.skeleton("Foo",
                      example_code=FALSE,
                      attributes=TRUE,
                      cpp_files=c("./Foo_R_wrapper.cpp", "./Foo.h", "/Foo.cpp")
                     )    

并在Bash

#tell the compiler to use C++11
export PKG_CXXFLAGS='`Rscript -e "Rcpp:::CxxFlags()"` -std=c++11'

#compile as usual
cd ./Foo/;
R -e 'Rcpp::compileAttributes(".",verbose=TRUE)';
cd ..;

R CMD build Foo;
#ensure that there are no bugs
R CMD check Foo;

其中Foo_R_wrapper.cpp包装了分别在Foo.hFoo.cpp中定义和实现的 C++ 函数 FooBar 。

Foo_R_wrapper.cpp可以包含以下内容:

#include <Rcpp.h>                                   

#include "Foo.h"                                    

// [[Rcpp::export]]                                 
SEXP FooBar(SEXP baa)
{                                                   
    Rcpp::NumericVector baa_vec(baa)
    //Do something from Foo.h (to baa_vec?)
    ...
    //etc
    return baa_vec;
}
于 2014-02-12T07:10:19.807 回答