为什么NDK在Build上抛出Undefined Reference错误?

时间:2015-02-18 08:10:42

标签: c++ c android-ndk

当我尝试使用ndk构建库时,我遇到了这个错误:

  

$ /cygdrive/d/android-ndk-r10d/ndk-build.cmd   [armeabi] SharedLibrary:libgame.so

     

jni / core / src / coreApplication.cpp:9:错误:未定义引用' CCoreMessaging :: CCoreMessaging()'

     

jni / core / src / coreApplication.cpp:11:错误:未定义引用' CCoreScreenContainer :: CCoreScreenContainer()'

     

jni / core / src / coreReference.h:75:错误:未定义引用' CCoreRefClass :: _ RemoveRef(void *,bool)'

     

jni / core / src / coreReference.h:76:错误:未定义引用' CCoreRefClass :: _ AddRef(void *,bool)'

     

jni / core / src / coreReference.h:58:错误:未定义引用' CCoreRefClass :: _ RemoveRef(void *,bool)'

     

jni / core / src / coreReference.h:58:错误:未定义引用' CCoreRefClass :: _ RemoveRef(void *,bool)'

     

jni / core / src / coreApplication.h:12:错误:未定义引用' CCoreMessaging :: ~CCoreMessaging()'

     

jni / core / src / coreReference.h:58:错误:对CCoreRefClass的未定义引用:: _ RemoveRef(void *,bool)'

     

collect2.exe:错误:ld返回1退出状态

     

make.exe:*** [obj / local / armeabi / libgame.so]错误1

coreApplication.h:

<other includes>
#include "coreResources.h"
#include "coreMessaging.h"

class CCoreApplication
{
public:
CCoreApplication(CCoreContext& a_context);
virtual ~CCoreApplication() {} 

CCoreScreenContainer& GetScreenContainer() { return *m_screenContainer; }
CCoreMessaging& GetMessages() { return m_messages; }

<other functions and properties>
CCoreMessaging m_messages;
};

coreApplication.cpp

#include "coreApplication.h"
#include "coreMessaging.h"
#include "coreScreenContainer.h"
<other includes>

CCoreApplication::CCoreApplication(CCoreContext& a_context):
 m_context(a_context)
{
m_screenContainer = new CCoreScreenContainer;
GetScreenContainer().SetBounds(CRectangle(0,0, a_context.GetPlatformHandler()->GetResolutionWidth(), a_context.GetPlatformHandler()->GetResolutionHeight()));
}

coreMessaging.h:

class CCoreMessaging
{
public:
CCoreMessaging(); 
~CCoreMessaging();
void SendMessage(int a_messageID, void* argument0=0, void* argument1=0);
void AddMessageHandler(CCoreMessageHandle& handle, CCoreMessageDelegate handler);
void RemoveMessageHandler(CCoreMessageHandle& handle);
protected:
typedef std::map<int, CCoreMessageDelegate> Map;
Map m_Delegates;
int m_AutoIncrement;
int m_SelfID;
static int m_SelfIDIncrement;
};

coreMessaging.cpp

#include "coreMessaging.h"
<other functions>

CCoreMessaging::CCoreMessaging() : m_AutoIncrement(1)
{
m_SelfID = m_SelfIDIncrement++;
gMessaging[m_SelfID] = this;
if(m_SelfID > 127)
    throw std::runtime_error("too many messaging instances");
}

coreReference.h

class MFAPI CCoreRefClass
{
public:
virtual void Delete() { delete this; }

void _AddRef(void* cls, bool strong);
void _RemoveRef(void* cls, bool strong);

void _IncRef();
bool _DecRef(); // returns true if reference decrease has lead to destruction. false if object is still valid
protected:
CCoreRefClass();
virtual ~CCoreRefClass();

typedef CCoreRefPtr<CCoreRefClass,false> RefType;
std::vector<RefType*> m_refData;

int m_refCount;
};

coreReference.cpp

#include "coreReference.h"

<other functions>
void CCoreRefClass::_RemoveRef( void* cls, bool strong )
{
RefType* refCls = (RefType*)cls;
int lastIdx=(int)m_refData.size()-1;
int refIdx = refCls->m_arrayIndex;
refCls->m_arrayIndex = -1;

// swap erase reference
m_refData[refIdx] = m_refData[lastIdx];
m_refData[refIdx]->m_arrayIndex = refIdx;
m_refData.pop_back();

if(strong)
    _DecRef();
}

void CCoreRefClass::_AddRef( void* cls, bool strong )
{
if(strong)
    _IncRef();

RefType* refCls = (RefType*)cls;
m_refData.push_back(refCls);
refCls->m_arrayIndex = (int)m_refData.size()-1;
}

这些错误中的大多数似乎都是由拼写错误产生的,但是这段代码确实在visual studio中完全编译为lib。 我希望这只是我错过或无法看到的东西。 如果需要,请随时提出更多要求。

编辑: application.mk

APP_STL := gnustl_static
APP_CPPFLAGS += -std=c++11
NDK_TOOLCHAIN_VERSION := 4.8
APP_PLATFORM := android-14

android.mk

LOCAL_PATH:=$(call my-dir)
include $(CLEAR_VARS)     
# Here we give our module name and source file(s)

LOCAL_MODULE    := game
LOCAL_SRC_FILES := coreBridge.cpp core/game.cpp core/src/coreApplication.cpp
LOCAL_LDLIBS :=  -llog -lGLESv2

include $(BUILD_SHARED_LIBRARY)
#include $(BUILD_EXECUTABLE)

2 个答案:

答案 0 :(得分:2)

感谢Bogdan V.我发现我需要包含所有源文件。 但不是将它们全部分开链接,而是一次全部链接:

Android.mk:

LOCAL_PATH:=$(call my-dir)
include $(CLEAR_VARS)

# Here we give our module name and source file(s)

LOCAL_MODULE    := libgame
LOCAL_SRC_FILES := coreBridge.cpp core/game.cpp

FILE_LIST := $(wildcard $(LOCAL_PATH)/../../../core/src/*.cpp)
LOCAL_SRC_FILES += $(FILE_LIST:$(LOCAL_PATH)/%=%) LOCAL_C_INCLUDES += $(LOCAL_PATH)/../../../core/include

LOCAL_LDLIBS :=  -llog -lGLESv2
LOCAL_CPP_FEATURES += exceptions
LOCAL_SHARED_LIBRARIES := libgame.so

include $(BUILD_SHARED_LIBRARY)
#include $(BUILD_EXECUTABLE)

其中:

FILE_LIST := $(wildcard $(LOCAL_PATH)/../../../core/src/*.cpp)
LOCAL_SRC_FILES += $(FILE_LIST:$(LOCAL_PATH)/%=%) LOCAL_C_INCLUDES += $(LOCAL_PATH)/../../../core/include

重要的是将它们全部联系起来。 这让我可以创建.so。

答案 1 :(得分:1)

您没有在构建中添加定义这些符号的文件:

coreMessaging.cpp
coreReference.cpp

想必

coreScreen.cpp (that i don't see here).

包含标题意味着只添加声明。您需要编译源以获取包含链接时用户的实际符号的目标文件。