编译Boost——Linux
作者:gale 日期:2008-09-16
Boost库简介
any - 类似于一个Varient。
array - STL风格封装下的定长数组。
bind - 实现函数对象的任意组合,是标准库中compose的超强扩展版。
call_traits - 实现自动判断传入参数的方式。
compose - bind的一个子集。
compressed_pair - 针对pair当中空成员做了一些优化。
concept_check - STL中做concept check的工具,单独实现成一个库。
conversion - 各种类型间的转化,lexical_cast和polymorphic_cast都很有用。
crc - 就是crc计算用的。
date_time - 有关日期和时间的内容。
dynamic_bitset - 标准库中bitset的动态长度版本。
filesystem - 提供了跨平台的文件操作,目录操作等。
format - 类似于C语言中经典的printf格式,但是加强了类型检查。
function - 用标准C++实现的delegate。
functional - 稍微增强了标准库中的bind1st一类函数。
graph - 非常不错的图库。将线性的STL扩展到平面中。
integer - 能够帮助简化对整数类型的处理。
interval -
io_state_saver - 简化IO操作。
lambda - 用标准C++模拟无名函数对象。
thread - 跨平台的线程实现。
相对于Windows来,Linux下的boost编译简单至极。没有那么多的可选编译器,没有那长的编译时间,没有那么多的硬盘使用量,统一的inlude和lib目录,你熟悉命令行,不使用IDE,不需要我那么罗嗦的介绍怎么配置EditPlus。
首先是下载boost,可以在此
提醒:做这些事情的时候你需要有root权限。
进入boost目录:
首先我们要编译bjam:
很快编译结束,默认情况下,bjam会被复制到/usr/local/bin/bjam。
现在你可以使用bjam编译boost了。
编译时间不会如windows那么长久,在我的电脑上编译了大约40分钟。你可以在前后使用df命令检查下磁盘使用,在我的电脑上,编译boost花费了500M的空间。
使用install会把头文件复制到/usr/local/include/boost-1_33_1中,把生成的lib复制到/usr/local/lib中。这些完成之后,记得要使用ldconfig来更新动态链接库。
在测试两个例子之前,我们先设置几个环境变量。
为了使其能够在登录时自动导入,你可以写一个脚本:
将其保存为/etc/profile.d/boost.sh,并使用chmod a+x boost.sh设置执行权限。
现在我们可以写两段代码来测试了。
第一个测试文件是lex.cpp:
运行:
输出:
你可以将$BOOST_ROOT改为$BOOST_INCLUDE,如果你没有设置环境变量,可以改为/opt/boost_1_33_1或者/usr/local/include/boost-1_33_1。
我们的第二个例子是re.cpp:
编译:
运行:
输出:
这里要使用-l指定了链接库。
现在boost的基本安装配置已经完成,但是我们可以再改进下。
如果不想每次都指定boost头文件目录,可以将其link到/usr/include中:
或者:
如果你依然嫌boost编译后占用的空间太大,可以在boost目录下使用bjam 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/local/bin/gccboost,赋予执行权限。
使用方法:
注意:使用此命令假设boost头文件在/usr/include中,如果假设不成立,请自行修改脚本此行:
为之前的注释行:
如若BOOST_ROOT和BOOST_LIB环境变量不存在,修改下面两行代码:
为之后注释行:
另外,gccboost将会自动修改输出的文件名为*.cpp的文件名(如lex.cpp将输出lex),如果不需要,请将下面的代码:
改为:
首先是下载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)