Включение Rcpp в проект Qt приводит к ошибке, влияющей на различные включенные библиотеки

Я делаю статическую библиотеку с Qt Creator, эта библиотека будет использоваться в более крупном приложении на основе графического интерфейса. Раньше я использовал Rcpp, но в качестве отправной точки я использовал R - так что, перемещая данные R в функции C++, чтобы что-то делать и получать результат, я привык делать это с R studio. На этот раз я начинаю в проекте Qt, и мне нужна таблица для хранения некоторых данных, и Rcpp::Dataframe прекрасно подойдет для работы, как и некоторые векторные классы. В моем исходном файле.cpp у меня есть оператор включения #include <Rcpp.h> и я посмотрел в Интернете и нашел пример использования библиотек Rcpp / RInside в Qt - хотя это больше о RInside, чем Rcpp: RInside и Qt

Кажется, что-то нужно добавить в файл.pro, чтобы заставить qmake делать правильные вещи. Я включил то, что, по моему мнению, необходимо для Rcpp, в свой файл.pro, прикрепленный ниже. Проблема, с которой я столкнулся, заключается в том, что до включения Rcpp в файл.pro и оператора include в файл.cpp, включение библиотек bpp, которые вы также можете увидеть в файле.pro ниже, не приводит к любые ошибки или предупреждения в компиляции. После включения операторов для включения R и Rcpp в файл.pro я получаю много предупреждений и одну ошибку, связанную с компонентом bpp, а именно: /local/yrq12edu/local/bpp/dev/include/Bpp/Numeric/NumConstants.h:96: error: expected unqualified-id before numeric constant, Есть также много Wunused-parameter сообщения, опять же, большинство из них, похоже, приходят из файлов внутри bpp. Это первый раз, когда мне приходилось иметь дело с чем-то вроде этого, и хотя я прочитал много документации по Qt, я все еще очень свежий новичок в Qt. Почему раньше я не получал ошибок, связанных с bpp, но получил их сейчас, когда попробовал включить Rcpp в свой проект Qt? И что я должен сделать, чтобы предпринять шаги, чтобы решить это?

#-------------------------------------------------
#
# Project created by QtCreator 2014-09-05T23:27:21
#
#-------------------------------------------------

QT       -= core gui

TARGET = libHybRIDS
TEMPLATE = lib
CONFIG += staticlib

SOURCES += hybridsengine.cpp
HEADERS += hybridsengine.h

INCLUDEPATH += /local/yrq12edu/local/bpp/dev/include
LIBS += -L/local/yrq12edu/local/bpp/dev/lib -lbpp-core
LIBS += -L/local/yrq12edu/local/bpp/dev/lib -lbpp-seq

# Set R home directory
R_HOME =                $$system(R RHOME)
# Set Rcpp include and lib flags.
RCPPINCL =              $$system($$R_HOME/bin/Rscript -e \'Rcpp:::CxxFlags\(\)\')
RCPPLIBS =              $$system($$R_HOME/bin/Rscript -e \'Rcpp:::LdFlags\(\)\')
# Set R cpp and ld flags. Also for BLAS and LAPACK it uses.
RCPPFLAGS =             $$system($$R_HOME/bin/R CMD config --cppflags)
RLDFLAGS =              $$system($$R_HOME/bin/R CMD config --ldflags)
RBLAS =                 $$system($$R_HOME/bin/R CMD config BLAS_LIBS)
RLAPACK =               $$system($$R_HOME/bin/R CMD config LAPACK_LIBS)

RCPPWARNING =           -Wno-unused-parameter

QMAKE_CXXFLAGS +=       $$RCPPWARNING $$RCPPFLAGS $$RCPPINCL
QMAKE_LFLAGS +=         $$RLDFLAGS $$RBLAS $$RLAPACK $$RCPPLIBS


unix {
    target.path = /usr/lib
    INSTALLS += target
}

РЕДАКТИРОВАТЬ: Следуя предложениям, я использовал пример файла pro в RInside/examples/qt в качестве отправной точки - сохраняя порядок операторов:

#-------------------------------------------------
#
# Project created by QtCreator 2014-09-05T23:27:21
#
#-------------------------------------------------

QT       -= core gui

TARGET = libHybRIDS
TEMPLATE = lib
CONFIG += staticlib

SOURCES += hybridsengine.cpp
HEADERS += hybridsengine.h

## comment this out if you need a different version of R,
## and set set R_HOME accordingly as an environment variable
R_HOME =        $$system(R RHOME)
#message("R_HOME is" $$R_HOME)

## include headers and libraries for R
RCPPFLAGS =         $$system($$R_HOME/bin/R CMD config --cppflags)
RLDFLAGS =      $$system($$R_HOME/bin/R CMD config --ldflags)
RBLAS =         $$system($$R_HOME/bin/R CMD config BLAS_LIBS)
RLAPACK =       $$system($$R_HOME/bin/R CMD config LAPACK_LIBS)

## if you need to set an rpath to R itself, also uncomment
RRPATH =        -Wl,-rpath,$$R_HOME/lib

## include headers and libraries for Rcpp interface classes
## note that RCPPLIBS will be empty with Rcpp (>= 0.11.0) and can be omitted
RCPPINCL =      $$system($$R_HOME/bin/Rscript -e \"Rcpp:::CxxFlags\(\)\")
RCPPLIBS =      $$system($$R_HOME/bin/Rscript -e \"Rcpp:::LdFlags\(\)\")

## for some reason when building with Qt we get this each time
##   /usr/local/lib/R/site-library/Rcpp/include/Rcpp/module/Module_generated_ctor_signature.h:25: warning: unused parameter ‘classname
## so we turn unused parameter warnings off
## no longer needed with Rcpp 0.9.3 or later
#RCPPWARNING =      -Wno-unused-parameter

## include headers and libraries for RInside embedding classes
RINSIDEINCL =       $$system($$R_HOME/bin/Rscript -e \"RInside:::CxxFlags\(\)\")
RINSIDELIBS =       $$system($$R_HOME/bin/Rscript -e \"RInside:::LdFlags\(\)\")

## compiler etc settings used in default make rules
QMAKE_CXXFLAGS +=   $$RCPPWARNING $$RCPPFLAGS $$RCPPINCL $$RINSIDEINCL
QMAKE_LIBS +=           $$RLDFLAGS $$RBLAS $$RLAPACK $$RINSIDELIBS $$RCPPLIBS

## addition clean targets
QMAKE_CLEAN +=      qtdensity Makefile

unix {
    target.path = /usr/lib
    INSTALLS += target
}

Помещение в мои операторы для включения библиотек bpp до или после операторов, относящихся к RInside и Rcpp, вызывает ту же ошибку, что и ранее.

Содержимое файла.h из bpp:

#ifndef _NUMCONSTANTS_H_
#define _NUMCONSTANTS_H_

#include <cmath>
#include <limits>

namespace bpp {

  /**
   * @brief this static class contains several useful constant values.
   *
   * This classe uses function in order to avoid the infamous "static initialization order fiasco".
   * C++0x solves this...
   */
  class NumConstants
  {

  public:
    /**
     * @name Golden ratio.
     *
     * The golden ratio, @f$\phi@f$ is equal to @f$\frac{1+\sqrt{5}}{2} = 1.6180339887498948482\ldots@f$.
     * We also define @f$R=\phi-1@f$ and @f$C = 1 - R@f$.
     * @{
     */
    static double GOLDEN_RATIO_PHI() { return (1. + sqrt(5.)) / 2.; }
    static double GOLDEN_RATIO_R() { return GOLDEN_RATIO_PHI() - 1.; }
    static double GOLDEN_RATIO_C() { return 1. - GOLDEN_RATIO_R(); }

    /** @} */

    static double MEGA() { return 1e6; }
    static double KILO() { return 1e3; }
    static double DECI() { return 1e-1; }
    static double CENTI() { return 1e-2; }
    static double MILLI() { return 1e-3; }
    static double MICRO() { return 1e-6; }
    static double NANO() { return 1e-9; }
    static double PICO() { return 1e-12; }

    static double SMALL() { return 1e-6; }
    static double TINY() { return 1e-12; }
    static double VERY_TINY() { return 1e-20; }
    static double VERY_BIG() { return static_cast<double>(1.7E+23); }

    /**
     * @name Define those constants in case they would not be available in stl/limits.
     *
     * @{
     */
    static double INF() { return std::numeric_limits<double>::has_infinity ? -log(0) : std::numeric_limits<double>::max(); }
    static double PINF() { return std::numeric_limits<double>::has_infinity ? -log(0) : std::numeric_limits<double>::max(); }
    static double MINF() { return std::numeric_limits<double>::has_infinity ? log(0) : std::numeric_limits<double>::min(); }
    static double NaN() { return NAN; }
    /** @} */

    static double PI() { return 3.141593; }
  };

}//end of namespace bpp.

#endif  //_NUMCONSTANTS_H_

1 ответ

Вот предположение: в R_ext/Constants.h, у нас есть:

trunk/src/include/R_ext/Constants.h
28:#define M_PI 3.141592653589793238462643383279502884197169399375
32:#define PI             M_PI

Это макроопределение сталкивается с определением функции для PI, Так что вы ожидаете, например,

#include <R.h> // pulls in 'R_ext/Constants.h'
#include "Bpp stuff" // definition for function named PI collides with macro

даст потенциальные ошибки. Определение макроса R для PI протекает в Bppи, следовательно, после того, как препроцессор сделан, эта строка:

static double PI() { return 3.141593; }

будет выглядеть

static double 3.141592653589793238462643383279502884197169399375() { return 3.141593; }

и компилятор может только сказать "Ват". Вот почему порядок включения может быть важным - в том числе Bpp Во-первых, можно избежать макро-загрязнения.

Похоже, что этого особого загрязнения можно избежать, определив #define STRICT_R_HEADERS (поскольку это то, что входит в запись).

Другие вопросы по тегам