链接器在编译时找不到重载运算符<<。C++

Linker can't find overloaded operator<< while compiling. C++

提问人:Youssof. K. 提问时间:4/15/2022 最后编辑:user12002570Youssof. K. 更新时间:11/10/2023 访问量:156

问:

我正在努力实现一个模板化的堆类。到目前为止,它真的很艰难。 我正在尝试显示我的堆,但收到此错误:

Undefined symbols for architecture x86_64:
  "operator<<(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, Heap<int> const&)", referenced from:
      _main in main.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

我在我的类中使用了模板,所以我的函数的实现也在头文件中,这里是 hpp 文件: (!):无需读取整个文件,只需 ostream& operator<<()。

/*
 Header file for Heap implemenetation
 */

#ifndef Heap_hpp
#define Heap_hpp

#include <stdio.h>
#include <vector>
#include <iterator>
#include <string>
#include <iostream>



template<typename elementType>
class Heap{
private:
    std::vector<elementType> myVecrtor;
    int mySize = 1; //The minimum size is 1 since the first element is a dummy.
    void perculateDown(int root);
    void perculateUp();
    
public:
    Heap();
    bool checkEmpty();
    void insert(elementType iteam);
    int getSize();
    std::vector<elementType> getHeap();
    elementType getMax();
    int getMaxIndex();
    void removeMax();
    void removeAtIndex(int index);
    friend std::ostream& operator<<(std::ostream &out, Heap const &h);
    
};

template<typename elementType>
Heap<elementType>::Heap(){
}

template <typename elementType>
bool Heap<elementType>::checkEmpty(){
    if(mySize > 1)
        return true;
    else
        return false;
}

template <typename elementType>
void Heap<elementType>::insert(elementType iteam){
    auto it = mySize;//iterator
    //myVecrtor.insert(it, iteam);//insert after first element(dummy)
    mySize++;
    perculateUp();
}

template <typename elementType>
std::vector<elementType> Heap<elementType>::getHeap(){
    return  myVecrtor;
}

template <typename elementType>
elementType Heap<elementType>::getMax(){
    return myVecrtor[getMaxIndex()];
}

template <typename elementType>
int Heap<elementType>::getMaxIndex(){
    int maxIndex = 1;
    elementType max = myVecrtor[maxIndex];
    for(int i = 0; i < myVecrtor.size(); i++){
        if(max < myVecrtor[i])
            maxIndex = i;
    }
    return maxIndex;
}

template <typename elementType>
void Heap<elementType>::removeMax(){
    int maxIndex = getMaxIndex();
    myVecrtor[maxIndex] = myVecrtor[mySize];
    mySize--;
    perculateDown(maxIndex);
}

template <typename elementType>
void Heap<elementType>::removeAtIndex(int index){
    myVecrtor[index] = myVecrtor[mySize];
    mySize--;
    perculateDown(index);
}

template <typename elementType>
std::ostream& operator<<(std::ostream &out, const Heap<elementType> &h){//it is giving me the error here
    out<<"\t\tHeap:";
    for(int i = 0; i < h.mySize; i++){
        out<<h.myVecrtor.at(i);
    }
    return out;
}

template <typename elementType>
void Heap<elementType>::perculateUp(){
    int loc = mySize - 1;
    int parent = loc /2;
   // while(parent >= 1 && myVecrtor[loc] > myVecrtor[parent]){
        elementType temp = myVecrtor[parent];
        myVecrtor[parent] = myVecrtor[loc];
        myVecrtor[loc] = temp;
        loc = parent;
        parent = loc / 2;
   // }
}

template <typename elementType>
void Heap<elementType>::perculateDown(int root){
    int r = root, c = r*2;
    while (r < mySize - 1) {
        if(c < mySize && myVecrtor[c] < myVecrtor[c+1])
            c++;
        if(myVecrtor[r] < myVecrtor[c]){
            elementType temp = myVecrtor[r];
            myVecrtor[r] = myVecrtor[c];
            myVecrtor[c] = temp;
            r = c;
            c *= 2;
        }
        else
            break;
    }
}


#endif /* Heap_hpp */

我尝试了所有其他功能,看看我是否犯了一个愚蠢的错误或其他什么,但除了这个之外,所有功能都有效: (!):ik 使用命名空间 std 是一个糟糕的 habbit,但我仅用于测试

#include <vector>
#include "Heap.hpp"
#include <ostream>
using namespace std;

int main(int argc, const char * argv[]) {
    Heap<int> h;
    h.insert(5);
    h.getHeap();
    h.getMaxIndex();
    h.getMax();
    h.removeMax();
    h.removeAtIndex(1);
    h.getHeap();
    cout<<h;
    
}

我不知道问题出在哪里,我发现与我的问题非常接近的一件事与命名空间 std 有关,但没有从中得到任何东西。 任何帮助将不胜感激!

C++ 运算符重载 std 显示

评论

0赞 PaulMcKenzie 4/15/2022
您是否知道 C++ 已经有堆函数

答:

2赞 user12002570 4/15/2022 #1

您当前拥有的好友声明是针对非模板的。operator<<operator<<

有 2 种方法可以解决这个问题:

方法 1

解决此问题,请将该友元声明替换为具有自己的参数子句的声明,以便您在类中有一个友元模板声明

template<typename elementType>
class Heap{
    //other code as before
    public:
        template<typename U>
        friend std::ostream& operator<<(std::ostream &out, Heap<U> const &h);
    
};

演示

方法 2

在这里,我们可以转发声明类模板和 .Heapoperator<<

//forward declaration for Heap<> 
template<typename T> class Heap;

//forward declaration for operator<<
template<typename T> std::ostream& operator<<(std::ostream& os, Heap<T> const &h);

template<typename elementType>
class Heap{
  //other members as before
    
public:

    friend std::ostream& operator<<<elementType>(std::ostream &out, Heap<elementType> const &h);
    
};

演示

评论

0赞 The Dreams Wind 4/15/2022
您当前拥有的运算符的友元声明<<是针对非模板运算符的<< - 前提是一个带有模板的类,这样的运算符什么时候可以工作?Heap
0赞 user12002570 4/15/2022
@TheDreamsWind 请参阅此现场演示,了解此类非模板的工作。operator<<
0赞 Youssof. K. 4/15/2022
我试过了,但后来它失败了;t 让我使用类 “h.mySize” 和 “h.myVector” 的私有成员,也不允许我查看我的公共函数,如 h.removeAtIndex()。
0赞 Youssof. K. 4/15/2022
注意:现在它能够找到它并调用它,但我不能使用上述成员。
0赞 user12002570 4/15/2022
@Youssof.K.我在回答中提供了 2 个现场演示。您是否单击它们并尝试过它们?它们有效,所以你不能说它不起作用。您可以将您的代码与这些现场演示相匹配,并找出问题所在。它们是您的代码的精确副本,只是我已经将我的答案中提到的解决方案应用于它们,以便您可以逐行并与您的代码匹配并找到问题。同样,请确保您使用的是而不是在所有头文件和源文件中。演示 1演示 2std::ostreamostream