c++ - 在构造函数/解构函数的情况下,“对 vtable 的 undefined reference ”

标签 c++ c++11 inheritance polymorphism

当我在带有 Release模式的 c++11 的 clang++ 中练习多态性时,我得到了 2 个“未定义对 vtable 的引用...”的错误。以下是我的部分代码。

#include <armadillo>
typedef float CHOSENONE;
typedef std::vector<std::vector<CHOSENONE> > fVect2d;
typedef std::vector<CHOSENONE> fVect;
class CfSubSet {
    private:
        fmat m_Data;
        frowvec m_Label;
        int m_NumInstances, m_NumFeatures ;
    public:
        CfSubSet(){
            cout<<"hello! \n" <<endl;
            m_NumInstances = 0;
            m_NumFeatures = 0;
            m_Data.zeros();
            m_Label.zeros();
        }
        CfSubSet(int iNumInstances, int iNumFeatures):m_NumInstances(iNumInstances),m_NumFeatures(iNumFeatures){
            m_Data.resize(iNumInstances,iNumFeatures);
            m_Label.resize(iNumInstances);
        }
        ~CfSubSet(){
            m_Data.reset();
            m_Label.reset();
        }
        inline int getNumInstances(){return m_NumInstances;};
        inline int getNumFeatures(){return m_NumFeatures;};
        inline fmat& getData(){return m_Data;};
        inline fmat& getLabel(){return m_Label;};
};

class CData
{
    protected:
        CfSubSet * m_Test;
        CfSubSet * m_Train;
    public:

        CData(){
            m_Test = new CfSubSet();
            m_Train = new CfSubSet();
        };

        ~CData(){
            delete m_Test;
            delete m_Train;
        };

        virtual void readDataFile(string ifFileTrainName) = 0;
        virtual void readLabelFile(string ifFileLabelFile) = 0;
        virtual void readTestFile(string ifFileTestFile) = 0;
        virtual void readGroundTruthFile(string ifGroundTruthFile) = 0;
};


class CMNISTSet : public CData
    {
    public:
        CMNISTSet() {};
        ~CMNISTSet(){};

        void readDataFile(string ifFileTrainName); 
        void readLabelFile(string ifFileLabelFile); 
        void readTestFile(string ifFileTestFile); 
        void readGroundTruthFile(string ifGroundTruthFile); 

    private:
        void ReadData(string ifFileName, fmat & rfOut); 
        void ReadLabel(string ifFileName, frowvec &rfOut); 
        int reverseInt (int i); 
    };

根据 Undefined reference to vtable

The solution is to ensure that all virtual methods that are not pure are defined. Note that a destructor must be defined even if it is declared pure-virtual

我已经实现了所有的虚函数,但是在析构函数的情况下,我没有明白这一点,因为我也定义了。我是否遗漏了我的代码中必须包含的任何内容?有没有人可以帮忙。提前谢谢你。

更新 1:我将完整的代码放在这里。在 MNISTSet.cpp

#include "MNISTSet.h"
int CMNISTSet::reverseInt (int i){
    unsigned char c1, c2, c3, c4;
    c1 = i & 255;
    c2 = (i >> 8) & 255;
    c3 = (i >> 16) & 255;
    c4 = (i >> 24) & 255;
    return ((int)c1 << 24) + ((int)c2 << 16) + ((int)c3 << 8) + c4;
}

void CMNISTSet::ReadData(string ifFileName, fmat & rfOut){
    ifstream ifs(ifFileName.c_str(),std::ios::in | std::ios::binary);
    int magic_number = 0;
    int number_of_images = 0;
    int rows = 0;
    int cols = 0;

    ifs.read((char*)&magic_number,sizeof(magic_number));
    magic_number= reverseInt(magic_number);
    ifs.read((char*)&number_of_images,sizeof(number_of_images));
    number_of_images= reverseInt(number_of_images);
    ifs.read((char*)&rows,sizeof(rows));
    rows= reverseInt(rows);
    ifs.read((char*)&cols,sizeof(cols));
    cols= reverseInt(cols);

    rfOut = fmat(number_of_images, rows*cols);
    cout << magic_number << " " << number_of_images << " " << rows << " " << cols << endl;

    for(int i = 0; i < number_of_images; i++){

        for(int row = 0; row < rows; row++){
            for(int col = 0; col < cols; col++){
                unsigned char temp = 0;
                ifs.read((char*)&temp,sizeof(temp));
                rfOut(i,rows*row+col) = (CHOSENONE)temp;
            }
        }
    }
}

void CMNISTSet::readDataFile(string ifFileTrainName){
    ReadData(ifFileTrainName, this.m_Train->getData());
}

void CMNISTSet::readTestFile(string ifFileTestFile){
    ReadData(ifFileTestFile, this.m_Test->getData());
}

void CMNISTSet::ReadLabel(string ifFileName, frowvec & rfOut){
    ifstream ifs(ifFileName.c_str(),std::ios::in | std::ios::binary);
    int magic_number = 0;
    int number_of_images = 0;

    ifs.read((char*)&magic_number,sizeof(magic_number));
    magic_number= reverseInt(magic_number);
    ifs.read((char*)&number_of_images,sizeof(number_of_images));
    number_of_images= reverseInt(number_of_images);

    rfOut = frowvec(number_of_images);
    cout << number_of_images << endl;

    for(int i = 0; i < number_of_images; i++){
        unsigned char temp = 0;
        ifs.read((char*)&temp,sizeof(temp));
        rfOut(i) = (CHOSENONE)temp;
    }
}

void CMNISTSet::readLabelFile(string ifFileTestFile){
    ReadLabel(ifFileTestFile, m_Labels);
}

void CMNISTSet::readGroundTruthFile(string ifGroundTruthFile){
    ReadLabel(ifGroundTruthFile, m_GroundTruth);
}

最佳答案

您的子类函数声明必须在某处定义。

为了测试,我将您的覆盖声明更改为:

void readDataFile(string ifFileTrainName) {}
void readLabelFile(string ifFileLabelFile) {}
void readTestFile(string ifFileTestFile) {} 
void readGroundTruthFile(string ifGroundTruthFile) {}

这有效。

如果您在单独的 .cpp 文件中定义函数。然后你需要指定他们的类。

void CMNISTSet::readDataFile( ... ){ ... }
...

关于c++ - 在构造函数/解构函数的情况下,“对 vtable 的 undefined reference ”,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29187827/

相关文章:

c++ - 没有#pragma comment lib 命令的 Unresolved external symbol 错误

c++ - C++ 中 f(g(), h()) 的求值顺序

c++ - 如何检查可能使用 SFINAE 的类型中是否存在字段?

c++ - 如何使用 std::bind() 调用基类版本的虚函数?

java - 覆盖java子类中的参数化构造函数

java - 实例化 T 的子类时如何返回泛型类型 T

c++ - 在 Visual Studio 中构建 wxWidgets-3.1.2 项目时如何修复 "Cannot find setup.h"错误?

c++ - Qt - 如何在 QLineEdit 中添加上标和下标?

c++ - 即使在 std::shared_ptr 拥有之后,shared_from_this 还是空的 _M_weak_this

c# - 通过继承实现接口(interface)