shamantou blog site

shamantou@gmail.com

导航

新编译的C++ Boost库

据说是c++未来的标准,一个相当强大的工具库,编译大概花费前后一小时时间,总计花费磁盘空间有5G不到,因为是默认的编译方式,所以debug,release版本全给我编译出来了。看了下打包文件中的文档资料还是相当丰富的。

为了方便查找,我把相关文档一起搬到了我的blog上,可以在首页找到对应的链接地址http://shamantou.vicp.cc/Ultilities/Boost

下面为编译的过程

Y:\boost1.34.1>bjam ^
More? --toolset=msvc stage
WARNING: No python installation configured and autoconfiguration
         failed.  See http://www.boost.org/libs/python/doc/building.html
         for configuration instructions or pass --without-python to
         suppress this message and silently skip all Boost.Python targets
warning: Python location is not configured
warning: the Boost.Python library won't be built
Building Boost.Regex with the optional Unicode/ICU support disabled.
Please refer to the Boost.Regex documentation for more information
(don't panic: this is a strictly optional feature).
...patience...
...patience...
...found 3734 targets...

最近公司里好像挺时髦这个boost库的,可能确实功能太强大的缘故吧,索性把它编译出来好好研究下吧。 希望和VTK一样,不是特别困难。

 

== follwing is added on 20090813

相对于Windows来,Linux下的boost编译简单至极。没有那么多的可选编译器,没有那长的编译时间,没有那么多的硬盘使用量,统一的inlude和lib目录,你熟悉命令行,不使用IDE,不需要我那么罗嗦的介绍怎么配置EditPlus。

首先是下载boost,可以在此

http://sourceforge.net/projects/boost

寻找一个合适的版本。比如我下载的是boost_1_33_1.tar.gz,解压到/opt。

 

tar xzvf boost_1_33_1.tar.gz -C/opt


提醒:做这些事情的时候你需要有root权限。

进入boost目录:

 

cd /opt/boost_1_33_1


首先我们要编译bjam:

 

cd tools/build/jam_src/
./build.sh


很快编译结束,默认情况下,bjam会被复制到/usr/local/bin/bjam。

现在你可以使用bjam编译boost了。

 

cd ../../..
bjam -sTOOLS=gcc install


编译时间不会如windows那么长久,在我的电脑上编译了大约40分钟。你可以在前后使用df命令检查下磁盘使用,在我的电脑上,编译boost花费了500M的空间。

使用install会把头文件复制到/usr/local/include/boost-1_33_1中,把生成的lib复制到/usr/local/lib中。这些完成之后,记得要使用ldconfig来更新动态链接库。

在测试两个例子之前,我们先设置几个环境变量。

 

BOOST_ROOT=/opt/boost_1_33_1
BOOST_INCLUDE=/usr/local/include/boost-1_33_1
BOOST_LIB=/usr/local/lib


为了使其能够在登录时自动导入,你可以写一个脚本:
 

#!/bin/sh#boost settings
BOOST_ROOT=/opt/boost_1_33_1
BOOST_INCLUDE
=/usr/local/include/boost-1_33_1
BOOST_LIB
=/usr/local/lib
  export BOOST_ROOT BOOST_INCLUDE BOOST_LIB


将其保存为/etc/profile.d/boost.sh,并使用chmod a+x boost.sh设置执行权限。

现在我们可以写两段代码来测试了。

第一个测试文件是lex.cpp:
 

#include <boost/lexical_cast.hpp>
#include 
<iostream>
int main()

   
using boost::lexical_cast;
    
int a = lexical_cast<int>("123");
    
double b = lexical_cast<double>("123.12");
    std::cout
<<a<<std::endl;
    std::cout
<<b<<std::endl;
    
return 0;
}

编译:
 

    g++ lex.cpp -I$BOOST_ROOT -o lex


运行:
 

    ./lex


输出:
 

    123
123.12


你可以将$BOOST_ROOT改为$BOOST_INCLUDE,如果你没有设置环境变量,可以改为/opt/boost_1_33_1或者/usr/local/include/boost-1_33_1。

我们的第二个例子是re.cpp:

 

#include <iostream>
#include 
<string>
#include 
<boost/regex.hpp>
int main()
 
{
   std::
string s = "who,lives:in-a,pineapple    under the sea?";
   boost::regex re(
",|:|-|\\s+");
   boost::sregex_token_iterator
     p(s.begin( ), s.end( ), re, 
-1);
   boost::sregex_token_iterator end;
   
while (p != end)
      std::cout 
<< *p++ << '\n';
}


编译:
 

    g++ re.cpp -I$BOOST_ROOT -lboost_regex-gcc -o re


运行:
 

    ./re


输出:
 

    who
lives
in
a
pineapple
under
the
sea?


这里要使用-l指定了链接库。

现在boost的基本安装配置已经完成,但是我们可以再改进下。

如果不想每次都指定boost头文件目录,可以将其link到/usr/include中:

 

    ln -s /opt/boost_1_33_1/boost /usr/include/boost


或者:

 

    ln -s /usr/local/include/boost-1_33_1/boost /usr/include/boost


如果你依然嫌boost编译后占用的空间太大,可以在boost目录下使用bjam clean:
 

    cd /opt/boost_1_33_1
bjam -sTOOLS=gcc clean


这个命令会清除编译时的中间文件,/usr/local/lib下带版本号的boost libs,和/usr/local/include下的boost头文件。但是同时节省了几百M的硬盘空间。

所以如果你使用了clean,记得将BOOST_INCLUDE更为BOOST_ROOT(/opt/boost_1_33_1),将 /usr/include/boost link到/opt/boost_1_33_1/boost,再有就是编译链接时的boost lib不要带版本号。

如果你觉得编译时手动链接敲那么长的名字比较麻烦,可以使用脚本来自动寻找链接:

 

#!/usr/bin/python
import os
import sys
import re
BOOST_ROOT 
= os.getenv('BOOST_ROOT')
BOOST_LIB 
= os.getenv('BOOST_LIB')
#BOOST_ROOT = '/opt/boost_1_33_1'
#
BOOST_LIB = '/usr/local/lib'
def getlibs():
    alls 
= os.listdir(BOOST_LIB)
    libpattern 
= re.compile(r'^libboost_([^-]+)-gcc')
    libs 
= {}
    
for lib in alls:
        m 
= libpattern.match(lib)
        
if m:
            libs[m.group(
1).lower()] = 1
    
return libs
pattern 
= re.compile(r'^\s*#include\s*<\s*boost/(.+)\.(h|hpp)\s*>')
libs 
= getlibs()
libskeys 
= libs.keys()
includes 
= {}
ENV 
= os.environ
ARGV 
= sys.argv[1:]
files 
= ARGV
if len(files) == 0: 
   sys.exit()
for f in files:
    
if f.lower().endswith('.cpp'):
        fp 
= open(f, 'r')
        lines 
= fp.readlines()
        
for ln in lines:
            m 
= pattern.match(ln)
            
if m:
                libname 
= m.group(1).lower()
                
if libname in libskeys:
                    includes[libname] 
= 1
libline 
= ' '.join(map(lambda lib: '-lboost_'+lib+'-gcc', includes.keys()))
obj 
= ARGV[0]
obj 
= obj[:len(obj)-4]
#cmd = 'g++ %s -I%s %s -o %s' % (' '.join(files), BOOST_ROOT, libline, obj)
cmd = 'g++ %s %s -o %s' % (' '.join(files), libline, obj)
print cmd
os.system(cmd)


将这段代码写进/usr/local/bin/gccboost,赋予执行权限。

使用方法:
 

    gccboost lex.cpp
gccboost re.cpp


注意:使用此命令假设boost头文件在/usr/include中,如果假设不成立,请自行修改脚本此行:
 

    cmd = 'g++ %s %s -o %s' % (' '.join(files), libline, obj)


为之前的注释行:
 

    cmd = 'g++ %s -I%s %s -o %s' % (' '.join(files), BOOST_ROOT, libline, obj)


如若BOOST_ROOT和BOOST_LIB环境变量不存在,修改下面两行代码:
 

    BOOST_ROOT = os.getenv('BOOST_ROOT')
BOOST_LIB = os.getenv('BOOST_LIB')


为之后注释行:
 

    BOOST_ROOT = '/opt/boost_1_33_1'
BOOST_LIB = '/usr/local/lib'


另外,gccboost将会自动修改输出的文件名为*.cpp的文件名(如lex.cpp将输出lex),如果不需要,请将下面的代码:
 

    cmd = 'g++ %s %s -o %s' % (' '.join(files), libline, obj)


改为:
 

    cmd = 'g++ %s %s' % (' '.join(files), libline)
<< 开始c#了一个未完成的工程 >>

发表评论:

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

最近发表

Powered By Z-Blog 1.8 Arwen Build 81206 Copyright 2006-2009 | ooplab.org | shamantou@gmail.com | 沪ICP备08011244号 | Some Rights Reserved.