【iOS & Mac】使用shell脚本批量合成多个.a文件为一个

先假设我有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

你可能感兴趣的:(【iOS & Mac】使用shell脚本批量合成多个.a文件为一个)