之前在编译jni时出现了undefined reference to function这个问题,查阅相关资料说是可能链接的库问题,然后通过各种gcc调整顺序解决了这个问题。这个工程是导师的工程,工程在Linux环境下运行无误,我从Linux环境往Android jni的一个迁移,由于文件的关联比较多代码量异常庞大和复杂(尝试着把方法直接复制到目标cpp中无果),而且我对gcc编译顺序这方面不是特别熟悉,在更改了Android.mk中的源文件顺序无果后,转向源码分析。
首先问题是出现在A.cpp中,其出错信息如下:
jni/A.cpp:518: error: undefined reference to ‘func1’
jni/A.cpp:531: error: undefined reference to ‘func2’
jni/A.cpp:533: error: undefined reference to ‘func3’
其中func1是在头文件B1.h中声明的,func2、func3是在头文件B2.h中声明的。
接着分析下A.cpp中的关键头文件包含信息:
//A.cpp
...
#include "A.h"
#include "X1.h"
#include "X2.h"
#include "X3.h"
...
然后是A.h中的头文件包含信息:
//A.h
...
#include "B1.h"
#include "B2.h"
...
发现X1.h、X2.h、X3.h中均包含了B1.h、B2.h这两个头文件
大概框架是这样的
A.cpp
- A.h
- B1.h
- B2.h
- X1.h
- B1.h
- B2.h
- X2.h
- B1.h
- B2.h
- X3.h
- B1.h
- B2.h
天知道他为什么能写出这样的代码……
猜想:A.cpp是否需要再次包含B1和B2两个头文件呢?于是改成了如下形式:
//A.cpp
...
#include "A.h"
#include "X1.h"
#include "X2.h"
#include "X3.h"
#include "B1.h"
#include "B2.h"
...
//A.h
...
#include "B1.h"
#include "B2.h"
...
尝试后发现依旧不行,于是再猜想:是否因为X1.h、X2.h、X3.h包含了B1.h、B2.h导致B1.h和B2.h实际上并没有包含进来呢?于是继续修改:
//A.cpp
...
#include "A.h"
#include "B1.h"
#include "B2.h"
...
//A.h
...
#include "B1.h"
#include "B2.h"
...
结果依旧是错误,那么又猜想,是否因为A.h包含了B1.h、B2.h导致A.cpp实际上并没有包含进这两个头文件呢,于是把A.h中的B1.h、B2.h删掉,改变如下:
//A.cpp
...
#include "A.h"
#include "B1.h"
#include "B2.h"
...
//A.h
...
//B1.h/B2.h去掉
...
发现编译居然没有报这个错了,但是报了其它函数缺失某个结构体的错误,这个结构体依赖于B1.h和B2.h,于是加上原来的X1.h、X2.h和X3.h,结构如下:
//A.cpp
...
#include "A.h"
#include "X1.h"
#include "X2.h"
#include "X3.h"
#include "B1.h"
#include "B2.h"
...
//A.h
...
//B1.h/B2.h去掉
...
发现编译居然通过了!
于是再次猜想:编译可能跟头文件的包含顺序有关,如果A.cpp先包含B1.h和B2.h这两个头文件是否能够获取到这两个头文件的func1、func2和func3呢?于是再作修改:
//A.cpp
...
#include "B1.h"
#include "B2.h"
#include "A.h"
#include "X1.h"
#include "X2.h"
#include "X3.h"
//#include "B1.h"
//#include "B2.h"
...
//A.h
...
#include "B1.h"
#include "B2.h"
...
居然还是编译成功了!
PS:C++菜鸟,对这个基本上不懂,只能一点点去尝试摸索。嵌套包含是个坑,Linux的编译和NDK的编译似乎也有些许不同,需要多尝试,多看前人经验,尽量少踩坑,节省时间啊~