如何解决将类声明放在.cpp文件中
| 是否可以在同一.cpp文件中包含类声明和实现? 我想在模拟对象的帮助下进行一些单元测试。这是我的测试的一些示例:// Some includes removed
#include \"abstractconnection.h\"
class ConnectionMockup : public AbstractConnection
{
Q_OBJECT
public:
explicit ConnectionMockup(QObject *parent = 0);
bool isReady() const;
void sendMessage(const QString &message);
void test_send_message(const QString &message);
bool ready;
QStringList messages;
};
ConnectionMockup::ConnectionMockup(QObject *parent)
: AbstractConnection(parent)
{
ready = true;
}
bool ConnectionMockup::isReady() const
{
return ready;
}
void ConnectionMockup::sendMessage(const QString &message)
{
messages.append(message);
}
void ConnectionMockup::test_send_message(const QString &message)
{
emit messageRecieved(message);
}
TestEmcProgram::TestEmcProgram(QObject *parent) :
QObject(parent)
{
}
void TestEmcProgram::open()
{
ConnectionMockup mockup;
EmcProgram program(&mockup);
QCOMPARE(...
...
...
如您所见,类ConnectionMockup仅由TestConnection类使用,而在其他任何地方都不需要它。因此,当我尝试编译该程序时,出现以下错误:
> testemcprogram.o: In function
> `ConnectionMockup\':
> /home/sasa/Desktop/QtPro/FocoKernel-build-desktop/../FocoKernel/testemcprogram.cpp:29:
> undefined reference to `vtable for
> ConnectionMockup\'
> /home/sasa/Desktop/QtPro/FocoKernel-build-desktop/../FocoKernel/testemcprogram.cpp:29:
> undefined reference to `vtable for
> ConnectionMockup\' testemcprogram.o: In
> function `~ConnectionMockup\':
> /home/sasa/Desktop/QtPro/FocoKernel-build-desktop/../FocoKernel/testemcprogram.cpp:14:
> undefined reference to `vtable for
> ConnectionMockup\'
是否可以在此处保留声明,或者我必须创建头文件并将声明移入该文件?
编辑:由于杰里·科芬先生(谢谢科芬先生)建议我可能未实现一些虚函数,因此我将在这里声明AbstractConnection,以便我们可以回顾这种可能性:
#include <QObject>
class AbstractConnection : public QObject
{
Q_OBJECT
public:
explicit AbstractConnection(QObject *parent = 0);
virtual ~AbstractConnection();
virtual bool isReady() const = 0;
signals:
void messageRecieved(const QString &message);
public slots:
virtual void sendMessage(const QString &message) = 0;
};
解决方案:感谢@ JCooper,@ iammilind和@Jerry Coffin,我们有了解决方案。从AbstractConnection中删除析构函数之后(因为它实际上不执行任何操作),并从ConnectionMockup中删除Q_OBJECT之后,它就可以工作了。
解决方法
是的,在一个文件中定义一个类及其成员函数是完全合法和允许的。实际上,从编译器的角度来看,基本上总是这样-您在标头中包含类定义,并将该标头包含在实现其成员函数的源文件中。
您遇到的错误看起来像链接器错误,而不是编译器错误。从您发布的内容来看,确切缺少的内容尚不完全清楚。一种可能是您的基类有一些纯虚函数,您未能在派生类中实现,但我完全不确定那是正确的。
,“ 3”宏声明了一组元对象成员函数。 MOC生成工具负责解析.h文件并定义这些函数声明。请注意,它不会解析.cpp文件。在您的情况下,找不到“ѭ4”,因为MOC工具没有解析您的.cpp文件。解决方案是将类定义移动到头文件中,并将头添加到.pro文件中。第二种解决方案(有点“ hacky”)是执行以下操作:
#include <QObject>
#include <QtDebug>
class Counter : public QObject
{
Q_OBJECT
public:
Counter() { value = 0; }
int getValue() const { qDebug() << \"getValue()\"; return value; }
public slots:
void setValue(int value);
signals:
void valueChanged(int newValue);
private:
int value;
};
#include \"main.moc\"
void Counter::setValue(int value)
{
qDebug() << \"setValue()\";
if (this->value != value) {
this->value = value;
emit valueChanged(value);
}
}
int main()
{
Counter a,b;
QObject::connect(
&a,&Counter::valueChanged,&b,&Counter::setValue);
a.setValue(12);
b.setValue(48);
return 0;
}
注意类定义下的“ #include \“ myfile.moc \”。
之所以可行,是因为qmake将使用#include指令在任何文件上调用MOC工具。因此,MOC将解析.cpp文件并生成元对象函数定义,从而解决了链接器错误。
,当基类具有任何不纯的virtual
函数时,编译最终二进制文件时必须包含其定义,否则会给出link4ѭ或typeinfo
的链接器错误。看下面的例子:
// Base.h
struct Base {
virtual void fun() = 0;
virtual ~Base();
};
// Base.cpp
#include\"Base.h\"
Base::~Base () {}
// Derived.cpp
#include\"Base.h\"
struct Derived : Base {
void fun () {}
};
int main () {
Derived d;
}
现在,Derived.cpp和Base.cpp的编译链接将正常工作。这两个.cpp文件也可以分别编译以创建目标文件,然后链接在一起。
从您的问题来看,我的感觉是,您并没有以某种方式附加class AbstractConnection
的.cpp / object文件,该文件仍然包含一个非纯虚函数-destructor
。如果您还与ConnectionMockup
一起编译该定义,则链接器错误应该不会出现。您可以编译包含析构函数主体的文件,也可以在类定义本身中定义析构函数主体。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。