简单介绍一下
最近工作中需要在IOS项目中使用一套由C++编写的数据传输组件, 该组件来源于旧的PC端软件,依赖BOOST库和C++版本的PROTOBUF库,在安卓项目可以直接通过JNI使用编译好的.O文件,由于IOS中不支持其他编译器编译的库, 因此需要在IOS环境中集成手机能用的相关静态库。相关环境配置比较麻烦且资料不多,经过了一番折腾才弄好,因此我在这里整理一番以供有需要的朋友参考。
一 在MAC上安装BOOST并打包支持iOS的静态库文件
这里我们是指定BOOST版本因此选择从官网下载手动配置的方式,这种方式比较通用。
- 先去官网下载你需要的BOOST版本这是地址
2.解压你下载到的文件
3.cd到该目录下并依次执行BOOST的编译指令sh bootstrap.sh sudo ./b2 headers sudo ./b2
过程比较慢 耐心等等
接下来要打包真机和模拟器能运行的静态库
这个脚本是从这篇文章复制并修改的
#!/bin/bash
# 请自行修改路径,cd到boost解压后的目录下
dir=`dirname $0`
cd "/..你的BOOST库路径../boost_1_65_0"
# 如果库文件已存在,直接退出
if [ -e ./stage/lib/libboost_date_time.a ]; then
echo "libraries exist. no need to build."
exit 0
fi
: ${COMPILER:="clang++"}
: ${IPHONE_SDKVERSION:=`xcodebuild -showsdks | grep iphoneos | egrep "[[:digit:]]+\.[[:digit:]]+" -o | tail -1`}
: ${XCODE_ROOT:=`xcode-select -print-path`}
: ${EXTRA_CPPFLAGS:="-stdlib=libc++ -std=gnu++11"}
echo "IPHONE_SDKVERSION: $IPHONE_SDKVERSION"
echo "XCODE_ROOT: $XCODE_ROOT"
echo "COMPILER: $COMPILER"
echo "bootstrap"
# 此脚本如果是被Xcode调用的话,会因为xcode export的某些变量导致失败,所以加了env -i。直接在命令行运行此脚本可以把env -i 去掉
env -i bash ./bootstrap.sh
echo "write project-config.jam"
# 默认生存的project-config.jam是编译Mac版的,这里直接调换掉
rm project-config.jam
cat >> project-config.jam <<EOF
using darwin : ${IPHONE_SDKVERSION}~iphone
: $XCODE_ROOT/Toolchains/XcodeDefault.xctoolchain/usr/bin/$COMPILER -arch armv7 -arch armv7s -arch arm64 $EXTRA_CPPFLAGS
: <striper> <root>$XCODE_ROOT/Platforms/iPhoneOS.platform/Developer
: <architecture>arm <target-os>iphone
;
using darwin : ${IPHONE_SDKVERSION}~iphonesim
: $XCODE_ROOT/Toolchains/XcodeDefault.xctoolchain/usr/bin/$COMPILER -arch i386 -arch x86_64 $EXTRA_CPPFLAGS
: <striper> <root>$XCODE_ROOT/Platforms/iPhoneSimulator.platform/Developer
: <architecture>ia64 <target-os>iphone
;
EOF
# 上面的代码里,两个using darwin分别是编译真机版和模拟器版的设置。每多一种CPU架构就要再加一个-arch xxx。
# 此处指定要编译的BOOST库
echo "build boost iphone dev"
#
./bjam -j16 --with-log --with-iostreams --with-regex --with-timer --with-exception --with-chrono --with-serialization --with-signals --with-date_time --with-filesystem --with-system --with-thread --build-dir=iphone-build --stagedir=iphone-build/stage toolset=darwin architecture=arm target-os=iphone macosx-version=iphone-${IPHONE_SDKVERSION} define=_LITTLE_ENDIAN link=static stage
echo "build boost iphone sim"
./bjam -j16 --with-log --with-iostreams --with-regex --with-timer --with-exception --with-chrono --with-serialization --with-signals --with-date_time --with-filesystem --with-system --with-thread --build-dir=iphonesim-build --stagedir=iphonesim-build/stage --toolset=darwin-${IPHONE_SDKVERSION}~iphonesim architecture=ia64 target-os=iphone macosx-version=iphonesim-${IPHONE_SDKVERSION} link=static stage cxxflags=-miphoneos-version-min=7.0
echo "lipo"
# 把各架构下的库文件合一,以便在xcode里可以少设置些搜索路径。做得更彻底些是各个分库合成一个大库。不过除非是把静态库加入到代码仓库,否则是浪费时间了。要合成的大库话请参考https://gist.github.com/rsobik/7513324原文。
mkdir -p stage/lib
lipo -create iphone-build/stage/lib/libboost_iostreams.a iphonesim-build/stage/lib/libboost_iostreams.a -output stage/lib/libboost_iostreams.a
lipo -create iphone-build/stage/lib/libboost_regex.a iphonesim-build/stage/lib/libboost_regex.a -output stage/lib/libboost_regex.a
lipo -create iphone-build/stage/lib/libboost_timer.a iphonesim-build/stage/lib/libboost_timer.a -output stage/lib/libboost_timer.a
lipo -create iphone-build/stage/lib/libboost_exception.a iphonesim-build/stage/lib/libboost_exception.a -output stage/lib/libboost_exception.a
lipo -create iphone-build/stage/lib/libboost_chrono.a iphonesim-build/stage/lib/libboost_chrono.a -output stage/lib/libboost_chrono.a
lipo -create iphone-build/stage/lib/libboost_serialization.a iphonesim-build/stage/lib/libboost_serialization.a -output stage/lib/libboost_serialization.a
lipo -create iphone-build/stage/lib/libboost_signals.a iphonesim-build/stage/lib/libboost_signals.a -output stage/lib/libboost_signals.a
lipo -create iphone-build/stage/lib/libboost_atomic.a iphonesim-build/stage/lib/libboost_atomic.a -output stage/lib/libboost_atomic.a
lipo -create iphone-build/stage/lib/libboost_wserialization.a iphonesim-build/stage/lib/libboost_wserialization.a -output stage/lib/libboost_wserialization.a
lipo -create iphone-build/stage/lib/libboost_date_time.a iphonesim-build/stage/lib/libboost_date_time.a -output stage/lib/libboost_date_time.a
lipo -create iphone-build/stage/lib/libboost_filesystem.a iphonesim-build/stage/lib/libboost_filesystem.a -output stage/lib/libboost_filesystem.a
lipo -create iphone-build/stage/lib/libboost_system.a iphonesim-build/stage/lib/libboost_system.a -output stage/lib/libboost_system.a
lipo -create iphone-build/stage/lib/libboost_thread.a iphonesim-build/stage/lib/libboost_thread.a -output stage/lib/libboost_thread.a
lipo -create iphone-build/stage/lib/libboost_log.a iphonesim-build/stage/lib/libboost_log.a -output stage/lib/libboost_log.a
# 库文件最终放在./stage/lib/下
echo "Completed successfully"
将这个脚本使用放在BOOST的目录下 cd到该目录并执行
会在该目录下得到一个 stage文件夹(如果要重新打包请先清空这个文件夹)
将得到的新的所有.a文件 替换到 /usr/local/lib文件夹下
然后删除所有libboost_xxxx.dylib文件 注意不要删非BOOST库的dylib文件
最后在项目中配置相关的路径就可以了
在 header search paths中添加 /usr/local/include
在 framework search paths 中添加 /usr/local/bin
在 other linker flags 指定你的项目用到的库
二 打包指定版本C++版的protobuf库
项目需要的PROTOBUF与本机安装版本不一致时需要将C++的对应版本PB项目打包成库集成到你的项目中,OC版的直接拖就行但是c++版本的比较麻烦,需要创建C++的项目并自行打包这里参考了这篇文章 。
- 下载你需要的版本的protobuf项目
2.cd到该项目下执行以下命令
./autogen.sh
3.在你的电脑上创建一个IOS 静态库项目
4.可以把自动生成的文件删掉
5.protobuf根目录/src/google/protobuf这个文件夹引入到你的项目中 (不要COPY IF NEED) 引用路径即可
6.在项目中搜索名称中带有"test"的文件并全部移除引用 移除"compiler"文件夹"testdata"文件夹 注意 移除项目引用即可不要原删除文件
7.CD到Protobuf根目录 文件夹下 执行 ./configure 在目录中生成config.h文件
- 将config.h文件拖入到项目中并将该文件设置为PCH文件
9.在项目中设置headers search path 为 src 文件夹的路径
10.把项目设置为release并RUN
就会得到对应真机或者模拟器的.A文件了 需要的话可以自己合并一下
11.使用时 将.A文件拖到你需要使用C++版PROTOBUF的项目中然后在header search path中添加 src文件夹的路径就可以了