先假设我有3个源文件:A、B、C,如果3个源文件分别编译成3个静态库,那么就会得到3个.a文件,分别是A.a、B.a和C.a;这样子在别人引用的时候需要导入的文件过多,那我们有没有什么办法可以将这些文件合并为一个lib.a文件呢?
注意,我这里说的合并多个库并不是指真机和模拟器版本的合并。
思路误区
很多人觉得既然.a就是一个静态库文件,那么是不是把多个静态库文件直接合成一个文件不就OK了?
但是,事实上这是不行的。因为静态库的模块是.o文件,你合并.a进去会导致无法找到符号,那么这个库就没用了。
所以,我们需要的是把.a里的.o提取出来,然后再把所有.o合并在一起,再打包成.a文件。
原理
首先,我们先认识下gcc的几个选项:
- -c表示只编译(compile)源文件但不链接,会把.c或.cc的c源程序编译成目标文件(二进制文件),一般是.o文件。
- -o用于指定输出(out)文件名。不用-o的话,一般会在当前文件夹下生成默认的a.out文件作为可执行程序。
- 在命令行中使用
ar rc lib.a lib.o
生成.a文件。
从这里可以看出,真正带符号的是.o文件,而.a文件是.o文件加层壳。
其实.a文件的结构和.tar文件没啥区别。我们可以用ar
命令去做操作:
-
x
命令解出来。 -
r
命令将文件插入库文件中。 -
c
命令建立库文件。 -
a
命令添加。 -
u
命令只将日期较新文件插入库文件中。
更多
rc
命令内容请参考该文章:linux ar命令用法
但是,单单使用ar
命令还是不够的。因为ar
命令只是把.o文件插入进去,并没有更新符号表,所以我们需要使用ranlib
命令更新符号表。
用法很简单,ranlib 库文件名
即可,如:ranlib lib.a
。
更多关于
ranlib
命令请参考文章:ranlib的作用
示例
假设我有上百个.a文件需要合并,如果在命令行自己敲,那不知道需要敲到什么时候了,所以我用shell来进行批处理。代码如下:
#!/bin/sh
#
# 批量合并多个.a文件
dir_path="${HOME}/Documents/library" # ${HOME}是登录用户的目录;这里先设置好需要解压的.a文件所在的绝对路径;根据实际情况修改
lib_path="${dir_path}/lib_folder" # 解压.a和合成.a绝对路径;根据实际情况修改
file_name="libxx.a" # 合成.a的名字;根据实际情况修改
#创建库文件目录
if [[ ! -d "${lib_path}" ]]
then
mkdir -p "${lib_path}" # 使用双引号防止存在空格导致错误
fi
cd "${lib_path}"
# 查找文件并解压
for file in $(ls "${dir_path}"|tr " " "?") # 解决名字带空格的问题
do
if [[ "${file}" =~ ".a" ]]
then
ar x "${dir_path}/${file}" #解压文件所在的路径 如果是在上级目录,可以用../${file}
fi
done
# 合并文件
ar cru "${file_name}" *.o
ranlib "${file_name}"
# 删除解压出来的文件
for file in $(ls|tr " " "?")
do
if [ "${file}" != "${file_name}" ] # 不是我们最终的.a文件,就删掉
then
rm -f "${file}"
fi
done
# 打开文件夹
open "${lib_path}"
在这里,我假设我的.a文件都放在dir_path
里,生成新的.a文件放在lib_path
里,新生成的.a名字叫file_name
。当然,这3个变量都需要你根据自己的实际情况去改。
lib_path
并不需要事先创建,我在脚本做了判断,如果没有该文件夹则会创建,并且包括其父文件夹也会创建。
如果脚本文件报没权限的错误,需要使用命令行来打开权限。
打开命令行,cd到脚本所在的目录,然后执行sudo chmod +x 脚本名
即可。
脚本我做了文件名带空格时的处理,所以无须担心文件名有空格会导致错误的问题。
iOS OC Swift Flutter开发群 139322447