我在linux中做了一个示例项目,但是运行主Makefile时出现错误

项目信息:

项目/数据库文件夹,具有文件database.h,database.cpp,bulid数据库,Makefile

数据库.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

Makefile
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;
}

Makefile
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

问题:数据库make文件运行正常,但主Makefile存在一些问题,例如

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

最佳答案

这行是错误的:

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

因为应该在该行上的对象main.o之后指定该库。这是由于链接器处理对象的方式。看这个例子:
gcc -o test someobject.o library.a

链接器将:
  • 查找someobject.o的所有 undefined reference ,并将它们存储为
  • 然后打开library.a并通过library.a解析 undefined reference
  • 然后关闭library.a

  • 如果对象和库相反,则链接器将打开library.a,并在其表中看不到 undefined reference 并将其关闭。然后,它尝试并编译someobject.o,并且永远不会满足 undefined reference

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

    07-24 09:45
    查看更多