在其中一个模板中使用头文件和 cpp 文件时编译时未定义引用

Undefined reference when compiling when using header and cpp file with templates in one of them

提问人:pdaranda661 提问时间:10/28/2021 更新时间:10/28/2021 访问量:342

问:

我一直在尝试编译我的项目,但在尝试时遇到了一些问题。具体出现的错误是:

[build] /usr/bin/ld: CMakeFiles/robot_control.dir/main.cpp.o:(.data.rel.ro._ZTVN4comm15cameraInterfaceE[_ZTVN4comm15cameraInterfaceE]+0x10): undefined reference to `comm::Interface<cv::Mat>::callbackMsg()'

我的项目现在组织如下:

-${HOME_WORKSPACE}
 |-main.cpp 
 |-src
   |-communication.cpp
   |-communication.hpp

头文件 (communication.hpp) 为:

#include <opencv2/opencv.hpp>
#include <gazebo/gazebo_client.hh>
#include <gazebo/msgs/msgs.hh>
#include <gazebo/transport/transport.hh>
#include <algorithm>

#ifndef COMM_GUARD
#define COMM_GUARD

namespace comm
{
    struct lidarMsg
    {
        float angle_min, angle_increment, range_min, range_max;
        int nranges, nintensities;
        std::vector<int> ranges;
    };

    template <typename T>
    class Interface
    {
        public:
            Interface() : received{false} {};

            virtual void callbackMsg();

            bool receptionAccomplished()
            {
                return this -> received;
            }
            
            T checkReceived()
            {
                return this -> elementReceived;
            }
        
        protected:
            bool received;
            T elementReceived;

    };

    class cameraInterface : public Interface<cv::Mat>
    {
        public:
            void callbackMsg(ConstImageStampedPtr &msg);
    };

    class lidarInterface : public Interface<lidarMsg>
    {
        public:
            void callbackMsg(ConstLaserScanStampedPtr &msg);
    };
}

#endif

源文件 (communication.cpp) 为:

#include <opencv2/opencv.hpp>
#include <algorithm>
#include <iostream>
#include "communication.hpp"

#ifndef COMM_CPP_GUARD
#define COMM_CPP_GUARD

namespace comm
{

    void cameraInterface::callbackMsg(ConstImageStampedPtr &msg)
    {
        std::size_t width = msg->image().width();
        std::size_t height = msg->image().height();
        const char *data = msg->image().data().c_str();
        cv::Mat im(int(height), int(width), CV_8UC3, const_cast<char *>(data));
        im = im.clone();
        cv::cvtColor(im, im, cv::COLOR_RGB2BGR);

        this->elementReceived = im;
        received = true;
    }

    void lidarInterface::callbackMsg(ConstLaserScanStampedPtr &msg) {

        this->elementReceived.angle_min = float(msg->scan().angle_min());
        this->elementReceived.angle_increment = float(msg->scan().angle_step());
        this->elementReceived.range_min = float(msg->scan().range_min());
        this->elementReceived.range_max = float(msg->scan().range_max());
        this->elementReceived.nranges = msg->scan().ranges_size();
        this->elementReceived.nintensities = msg->scan().intensities_size();

        for (int i = 0; i < this->elementReceived.nranges; i++)
        {
            if (this->elementReceived.ranges.size() <= i)
            {
                this->elementReceived.ranges.push_back(std::min(float(msg->scan().ranges(i)), this->elementReceived.range_max));
            }
            else
            {
                this->elementReceived.ranges[i] = std::min(float(msg->scan().ranges(i)), this->elementReceived.range_max);
            }
        }
    }
}

#endif

主文件(main.cpp)包含以下标头:

#include <gazebo/gazebo_client.hh>
#include <gazebo/msgs/msgs.hh>
#include <gazebo/transport/transport.hh>
#include <opencv2/opencv.hpp>
#include <opencv2/calib3d.hpp>
#include <iostream>
#include <stdlib.h>

#include "src/communication.hpp"

我包含了 #ifndef/#define/#endif 的一部分,因为它是我在其他问题中找到的解决此类问题的解决方案。我一直在切换CMakeLists.txt文件,但仍然没有解决方案可以解决此错误。

C++ 编译器错误 标头 undefined-reference

评论

0赞 user4581301 10/28/2021
无关:有史以来最好的凉亭使用。
2赞 Richard Critten 10/28/2021
这回答了你的问题吗?为什么模板只能在头文件中实现?

答:

1赞 Joseph Larson 10/28/2021 #1

你不能这么做:

        virtual void callbackMsg();

您必须实际提供 .h 文件中所有模板方法的实现。

评论

0赞 pdaranda661 10/28/2021
实际上,我刚刚实现了这一点,并且它正在起作用!我将重新检查一下这些修饰符。