主Makefile中的未定义引用

时间:2014-01-25 11:27:39

标签: c++ linux makefile

我在linux中做了一个示例项目,但是在运行main Makefile时遇到错误

项目信息:

项目/数据库文件夹包含文件database.h,database.cpp,bulid-database,Makefile

database.h

/*data base file*/
#include<iostream>
using namespace std;
class mydatabase
{
public:
    mydatabase(int a , int b);
    int sum(){return m_a +m_b;}
    int diff(){return m_a -m_b;}
    int mul(){return m_a *m_b;}
    float div(){return m_a /m_b;}
    int reminder(){return m_a %m_b;}

private:
    int m_a , m_b;
};

database.cpp

#include "database.h"
mydatabase::mydatabase(int a ,int b):m_a(a) , m_b(b)
{
}

bulid数据库

make
if [ -f libdatabase.a ];
then
   echo "Database-Library Build Success"
   cp libdatabase.a ../LIBs/
else
    echo "databse-Library Build Failure"
fi

生成文件

HEADERFILES = $(wildcard *.h)
CPPFILES = $(wildcard *.cpp)
OBJFILES = $(patsubst %.cpp,%.o ,$(wildcard *.cpp))
$(OBJFILES): %.o : %.cpp $(HEADERFILES)
    g++ -c -o $@ $<
    ar ruv libdatabase.a $@
    ranlib libdatabase.a

项目/主文件夹包含文件main.cpp,Makefile

的main.cpp

#include "database.h"
#include <iostream>
int main()
{
    mydatabase *obj = new mydatabase(10 ,5);
    std::cout<<"sum is"<<obj->sum()<<endl;
    std::cout<<"diff is"<<obj->diff()<<endl;
    std::cout<<"mul is"<<obj->mul()<<endl;
    std::cout<<"div is"<<obj->div()<<endl;
    std::cout<<"reminder is"<<obj->reminder()<<endl;
    getchar();
    return 0;
}

生成文件

CC        = g++
INCPATH  = -I. \
       -I.. \
       -I../database
LIBPATH  = -L../LIBs
LDFLAGS   = ${LIBPATH}/libdatabase.a
CFLAGS    = ${INCPATH} 
testdate:main.o
    $(CC) $(CFLAGS) -o testdate main.o $(LDFLAGS)
main.o:main.cpp
    $(CC) $(CFLAGS) -c -o main.o main.cpp

问题:数据库生成文件工作正常但主要的Makefile我遇到了一些问题,如

错误: main.o:在函数main': main.cpp:(.text+0x92): undefined reference to mydatabase :: mydatabase(int,int)中 collect2:ld返回1退出状态

2 个答案:

答案 0 :(得分:6)

这一行错了:

$(CC) $(CFLAGS) -o testdate $(LDFLAGS) main.o

因为库应该在行上的对象main.o之后进行特定。这是由于链接器处理对象的方式。看看这个例子:

gcc -o test someobject.o library.a

链接器将:

  1. 查找someobject.o的所有未定义引用并存储它们
  2. 然后打开library.a并通过library.a
  3. 解析未定义的引用
  4. 然后关闭library.a
  5. 如果对象和库是相反的,那么链接器打开library.a,在其表中看不到未定义的引用并关闭它。然后它尝试并编译someobject.o并且永远不会满足未定义的引用

    修改: 这是GCC的一个众所周知的警告,可以看到更详细的堆栈溢出说明here,选项--start-group--end-group可以帮助解决A依赖于B的情况,以及B取决于A.

答案 1 :(得分:1)

这是你的Makefile。你想要:

libdatabase.a

-ldatabase

main编译行的末尾