我可以只 automoc 一个 header 继承 QObject 的 object 吗?

Can I automoc a header only object that inherits QObject?

我是 Qt 的新手,我正在将它整合到我们的项目中。我做了一个非常小的 object ,它继承了我只是与计时器一起使用的 QObject 。我为它创建了一个仅 header 的文件,但很快意识到编译器不喜欢它。所以我为它创建了一个免费的 .cpp 文件来消除错误并且它似乎有效。

我的问题真的是,我可以创建一个 header 只有 object 继承 QObject 并允许它自动运行吗?或者我每次都需要创建一个免费的 cpp 文件吗?

我生成了少量代码来复制,这说明了我的意思。

CMakeLists.txt

cmake_minimum_required(VERSION 2.8.11)

# Standardly
set(CMAKE_CXX_STANDARD 11)

# Findn includes in corresponding directories?
set(CMAKE_INCLUDE_CURRENT_DIR ON)
# Instruct CMake to run moc automatically when needed
set(CMAKE_AUTOMOC ON)

find_package(Qt5Widgets)

add_executable(test Main.cpp)

target_link_libraries(test Qt5::Widgets)

Main.cpp

#include <QApplication>

#include "Header.h"

int main( int argc, char *argv[] )
{
    QApplication app( argc, argv );

    HeaderOnly();

    return( app.exec() );
}

Header.h

#ifndef HEADER_H
#define HEADER_H

#include <QObject>
#include <QTimer>
#include <QDebug>

class HeaderOnly : public QObject
{
    Q_OBJECT

    public:
        HeaderOnly() :
            timer_( new QTimer( this ) )
        {
            QObject::connect( timer_, SIGNAL( timeout() ), this, SLOT( timeout() ) );

            timer_->start( 1000 );
        }

    private slots:
        void timeout()
        {
            qDebug() << "Why hello!";
        }

    private:
        QTimer *timer_;
};

#endif

输出

$ make
[ 25%] Automatic MOC for target test
Generating MOC predefs moc_predefs.h
[ 25%] Built target test_autogen
Scanning dependencies of target test
[ 50%] Building CXX object CMakeFiles/test.dir/Main.cpp.o
[ 75%] Linking CXX executable test
CMakeFiles/test.dir/Main.cpp.o: In function `HeaderOnly::HeaderOnly()':
Main.cpp:(.text._ZN10HeaderOnlyC2Ev[_ZN10HeaderOnlyC5Ev]+0x32): undefined reference to `vtable for HeaderOnly'
CMakeFiles/test.dir/Main.cpp.o: In function `HeaderOnly::~HeaderOnly()':
Main.cpp:(.text._ZN10HeaderOnlyD2Ev[_ZN10HeaderOnlyD5Ev]+0xf): undefined reference to `vtable for HeaderOnly'
collect2: error: ld returned 1 exit status
make[2]: *** [CMakeFiles/test.dir/build.make:124: test] Error 1
make[1]: *** [CMakeFiles/Makefile2:68: CMakeFiles/test.dir/all] Error 2
make: *** [Makefile:84: all] Error 2

当然把这个Header.cpp加到源里就可以消除错误了:

#include "Header.h"

#include "moc_Header.cpp"

有两种解决方法:

  1. 在项目中显式包含头文件,就像在 .pro 文件中一样:

    add_executable(test "Main.cpp" "Header.h")
    
  2. #include "moc_Header.cpp" 添加到任何 一个 (并且只有一个)的 .cpp 文件中,例如在 Main.cpp:

    // Main.cpp
    ...
    #include "moc_Header.cpp"
    

    如果您在这样做时收到 CMP0071 政策警告 - 这是虚假的。您需要删除构建文件夹并重新运行 cmake 以重新配置构建。

在源列表中包含头文件:

add_executable(test Main.cpp Header.h)

这会让 CMake 知道您有一个带有 QObject 声明的头文件。所以它会自动计算出 (CMAKE_AUTOMOC ON) 它需要 运行 moc 并将其包含到构建中。