2013-03-26 47 views
18

我想运行程序,使用示例代码的Ubuntu 12.10 boost :: filesystem,但它不想构建。无法链接程序使用Boost.Filesystem

#include <iostream> 
#include <boost/filesystem.hpp> 
using namespace boost::filesystem; 
using namespace std; 

void fun(const string& dirPath); 
int main() 
{ 
    fun("/home"); 
    return 0; 
} 

void fun(const string& dirPath) 
{ 
    path p (dirPath); 

    if (exists(p)) 
    { 
     if (is_regular_file(p)) 
      cout << p << " size is " << file_size(p) << '\n'; 

     else if (is_directory(p))  
      cout << p << "is a directory\n"; 

     else 
      cout << p << "exists, but is neither a regular file nor a directory\n"; 
    } 
    else 
     cout << p << "does not exist\n"; 
} 

而且CMake的代码:

project(tttest) 
cmake_minimum_required(VERSION 2.8) 
aux_source_directory(. SRC_LIST) 
add_executable(${PROJECT_NAME} ${SRC_LIST}) 
FIND_PACKAGE(Boost 1.53 COMPONENTS filesystem system REQUIRED) 
include_directories(${Boost_INCLUDE_DIR}) 
link_directories(${Boost_LIBRARY_DIR}) 
TARGET_LINK_LIBRARIES(${PROJECT_NAME} ${Boost_LIBRARIES}) 

不幸的是它会产生错误

CMakeFiles/tttest.dir/main.cpp.o: In function `boost::filesystem::exists(boost::filesystem::path const&)': 
main.cpp:(.text._ZN5boost10filesystem6existsERKNS0_4pathE[_ZN5boost10filesystem6existsERKNS0_4pathE]+0x19): undefined reference to `boost::filesystem::detail::status(boost::filesystem::path const&, boost::system::error_code*)' 
CMakeFiles/tttest.dir/main.cpp.o: In function `boost::filesystem::is_directory(boost::filesystem::path const&)': 
main.cpp:(.text._ZN5boost10filesystem12is_directoryERKNS0_4pathE[_ZN5boost10filesystem12is_directoryERKNS0_4pathE]+0x19): undefined reference to `boost::filesystem::detail::status(boost::filesystem::path const&, boost::system::error_code*)' 
CMakeFiles/tttest.dir/main.cpp.o: In function `boost::filesystem::is_regular_file(boost::filesystem::path const&)': 
main.cpp:(.text._ZN5boost10filesystem15is_regular_fileERKNS0_4pathE[_ZN5boost10filesystem15is_regular_fileERKNS0_4pathE]+0x19): undefined reference to `boost::filesystem::detail::status(boost::filesystem::path const&, boost::system::error_code*)' 
CMakeFiles/tttest.dir/main.cpp.o: In function `boost::filesystem::file_size(boost::filesystem::path const&)': 
main.cpp:(.text._ZN5boost10filesystem9file_sizeERKNS0_4pathE[_ZN5boost10filesystem9file_sizeERKNS0_4pathE]+0x19): undefined reference to `boost::filesystem::detail::file_size(boost::filesystem::path const&, boost::system::error_code*)' 
collect2: error: ld returned 1 exit status 

这是什么问题的原因以及如何解决呢?

+2

尝试建立与自动检测增压版本:FIND_PACKAGE(升压COMPONENTS filesystem system REQUIRED) – podshumok 2013-03-26 12:07:44

+0

我向标签添加了cmake。这个问题似乎与cmake有关。 – drescherjm 2013-03-26 13:41:32

+1

您的CMake版本是否支持1.53版本?您可能想要打印出CMakeLists.txt中$ {Boost_LIBRARIES}包含的内容,以更好地理解正在发生的事情。 MESSAGE(状态$ {Boost_LIBRARIES})应该这样做。 – drescherjm 2013-03-26 13:44:44

回答

1

对于一些升压模块,您必须编译库链接它们(使用bootstrap.sh)。 在你的情况,你必须编译和链接Filesystem,并且probalbly System

看一看here

例如:

  • ./bootstrap.sh(bjam的)
  • rm -rf bin.v2 stage(between 2 bjam commands)
  • ./bjam release toolset = gcc address-model = 64 cxxflags = -fPIC
  • ./bjam调试工具集= GCC地址模式= 64个CXXFLAGS = -fPIC

如果您在Windows上链接,您不必手动链接你的图书馆,因为它们会自动使用编译链接。在Linux上,你必须这样做。

根据文档,这些模块需要你购买或修建图书馆:

  • Boost.Filesystem的
  • Boost.GraphParallel
  • 了Boost.Iostreams
  • Boost.MPI
  • 加速.ProgramOptions
  • Boost.Python
  • Boost.R egex
  • Boost.Serialization
  • Boost.Signals
  • Boost.System
  • Boost.Thread
  • Boost.Wave
+1

这是在Linux上,我希望通过ubuntu中的包管理器来安装。 – drescherjm 2013-03-26 13:42:52

22

提高文件系统的是,有一些ABI问题Boost库之一相对于由于C++ 0x或C++ 11引起的函数签名更改而言。CF升压票:https://svn.boost.org/trac/boost/ticket/6779

你有三种解决方法:

  1. 禁止C++ 11作用域的枚举关注升压包含在你的程序的头文件用#include(CF http://www.ridgesolutions.ie/index.php/2013/05/30/boost-link-error-undefined-reference-to-boostfilesystemdetailcopy_file/):

    #define BOOST_NO_CXX11_SCOPED_ENUMS 
    #include <boost/filesystem.hpp> 
    #undef BOOST_NO_CXX11_SCOPED_ENUMS 
    

    但是这个解决方案不是一个完整的解决方案,我读它并不适用于每个人。

  2. 构建BOOST与C++ 11的选项(你使用你的应用程序相同的选项):http://hnrkptrsn.github.io/2013/02/26/c11-and-boost-setup-guide

    我看也并不为大家工作。

  3. 设置专用于您的应用程序的交叉编译器,您可以在专用环境中重建所需的所有库。这确保了连贯性,稳定性和更多的可维护性,当然也是推荐的解决方案。 我还没有阅读,如果它已经过测试 - 可能是的,可能它的工作原理。无论如何,现在在计算机科学中,交叉编译已经很好掌握了。你会发现很多好的教程和支持。在Linux Gentoo中,他们拥有奇妙的sys-devel/crossdev软件包,使其变得非常简单。

在我自己的案例中,解决方案1已经解决了这个问题。一旦我遇到另一个,我将切换到解决方案3.所以,我还没有测试过它。

+0

上面只是一个改进,如果你正在使用libboost低于1.51的使用需要使用 的#define BOOST_NO_SCOPED_ENUMS 的#include“升压/ filesystem.hpp” 和#undef BOOST_NO_SCOPED_ENUMS – 2015-10-26 18:47:25

+2

注意,对于选项1,** **所有升压包括绝已经定义了'BOOST_NO_CXX11_SCOPED_ENUMS',否则肯定会跟踪ODR违规行为。 – ildjarn 2016-01-26 15:53:20

5

您需要在链接时添加libboost_filesystem库。或者如果你的应用程序是多线程的话,libboost_filesystem-mt。就像这样:

g++ -o file -lboost_filesystem-mt source_file.cpp 
+0

如果你使用的CMake是 'target_link_libraries(my_executable $ {Boost_LIBRARIES})' – Adrian 2017-04-26 10:38:53

0

您需要添加以下库:如果你使用一个Makefile

g++ -o file -lboost_system -lboost_filesystem sourcefile.cpp 

CC=gcc 
CXX=g++ 
PROG = program 
CXXFLAGS := -std=c++1y -g -Wall 
LDFLAGS = -L/usr/local/lib -L/usr/lib/x86_64-linux-gnu 
LIBS= -lboost_system -lboost_filesystem 

SRCS= main.cpp 

OBJS=$(subst .cpp,.o,$(SRCS)) 

all: $(OBJS) 
    $(CXX) $(CXXFLAGS) -o $(PROG) $(OBJS) $(LIBS) $(LDFLAGS)