记一次特殊的duplicate symbol报错,以及c++头文件找不到的报错

如题所述

第1个回答  2022-07-16
这次在集成第三方sdk的时候遇到的duplicate symbol报错十分诡异,当我的项目中有一个.mm文件的时候,再新建一个.mm文件都会有这样的报错。

一、通常一般遇到这种错误都是由一下这几点引起的:

二、针对上面的几点,分别有以下改正的方法:

针对上面的第四点,可以采取将多个库文件合并,或者将库文件中有重复的部门删除掉,只保留一份

比如我遇到了下面的错误:

1.根据错误的信息来看,两个静态库文件中有重复的类。我这里解决的办法是将其中一个静态库中把重复的类删除掉。

2.使用lipo -info命令来查看静态库支持的指令集

由于我现在xcode9.2打出的包都是 armv7 arm64的,所以这里只需要针对这两个指令集来分别操作

3.先将库文件拆分成不同指令集的

4.然后分别删除armv7指令集 和 arm64指令集下重复的文件
使用Ar -dv xxx(库文件) xxx.o(重复的文件) 命令来删除

想查看库文件中有哪些文件可以使用 ar - t 命令

5.合并静态库

然而我遇到的duplicate symbol错误都不是上面的原因造成的,最开始我在报错的类里面搜索“duplicate symbol _StartTime in”中的StartTime这个关键词的时候,没有搜索到有这个关键词。而且奇怪的是当我项目中有一个.mm文件的时候,再新建一个.mm文件都会有这样的报错。真是很诡异!

报错信息

在困扰了很久之后我开始在项目中搜索StartTime这个关键词,我发现在很多类中都有这个关键词,然后一个个的排查。终于在一个类中找到了一个可以的地方。有个相同名称的全局变量,在被其他类使用的时候,没有使用extern来声明可以被其他的类使用,最后在.h文件中在该变量前面加上FOUNDATION_EXTERN关键字后这个问题才被解决😄

此外,在这次集成三方sdk时候,由于三方sdk是用oc和c++混编的。集成的时候很容易报错,比如说下面这个报错:

报错的信息显示找不到c++文件"sstream"。首先明确的一点就是编译器会去识别.mm文件是oc与c++混编的,oc与c++混编不需要什么特殊的设置,在这里加载c++的头文件是没有问题的。但是需要注意一点的是在加在c++的库文件的时候,需要在所有使用到包含了这个头文件的类文件都要使用.mm后缀。
比如我项目中有一个叫tom的类,在该类中引用类头文件"sstream",这时候tom的.m文件后缀应该改为.mm。有一个jerry的类,引用了tom这个类,那么这时候jerry的.m文件应该改为.mm文件。
相似回答