Throw-catch вызывает ошибки компоновки - PullRequest
0 голосов
/ 07 мая 2009

Я получаю ошибки связывания следующего типа:

Festival.obj: ошибка LNK2019: Неразрешенный внешний символ "Общественность: void __thiscall Tree :: add (class Price &) " (? Добавить @? $ Tree @ VPrice @@@@ QAEXAAVPrice @@@ Z) ссылка в функции __catch $? AddBand @ фестиваль @@ QAE? AW4StatusType @@ HHH @ Z $ 0

Раньше я думал, что это связано с механизмом try-catch, но с тех пор было сказано иначе. Это обновленная версия вопроса.

Я использую Visual Studio 2008, но у меня похожие проблемы в g ++.

Соответствующий код:

В Festival.cpp

#include "Tree.h"
#include <exception>

using namespace std;

class Band{
public:
    Band(int bandID, int price, int votes=0): bandID(bandID), price(price), votes(votes){};
...
private:
...

};

class Festival{
public: 
    Festival(int budget): budget(budget), minPrice(0), maxNeededBudget(0), priceOffset(0), bandCounter(0){};
    ~Festival();
    StatusType AddBand(int bandID, int price, int votes=0);
    ...

private: 
    Tree<Band> bandTree;
    ...

};

StatusType Festival::AddBand(int bandID, int price, int votes){
    if ((price<0)||(bandID<0)){
        return INVALID_INPUT;
    }
    Band* newBand=NULL;
    try{
        newBand=new Band(bandID,price-priceOffset,votes);
    }
    catch(bad_alloc&){return ALLOCATION_ERROR;}
    if (bandTree.find(*newBand)!=NULL){
        delete newBand;
        return FAILURE;
    }
bandTree.add(*newBand);
....
}

В Tree.h:

template<class T>
class Tree{
public:
    Tree(T* initialData=NULL, Tree<T>* initialFather=NULL);
    void add(T& newData);
....
private:
....
};

Интересно, что у меня нет ошибок компоновки, когда я пытаюсь использовать функции дерева, когда тип T является примитивным типом типа int

Ответы [ 6 ]

2 голосов
/ 07 мая 2009

Есть ли Tree.cpp? Если есть, может быть, вы забыли связать это? Где реализация Tree :: add? Кроме того, я не вижу, где вы вызываете Tree :: add. Я думаю, это должно быть внутри оператора try, сразу после нового?

Просто напоминание:

Для большинства компиляторов (то есть тех, которые практикуют раздельную компиляцию) реализация функций-членов класса шаблона должна быть видимой во время компиляции исходного файла, который использует класс шаблона. Обычно люди следуют этому правилу, помещая реализацию функций-членов в заголовочный файл.

Может, Tree :: add находится не внутри заголовка? Тогда возможным решением в обсуждаемом случае будет поместить реализацию Tree :: add в заголовочный файл.

Разница между обычными классами и шаблонными классами существует, потому что шаблонные классы не являются "реальными" классами - это, ну, в общем, шаблон. Если бы вы определили свой класс Tree как обычный класс, компилятор мог бы использовать ваш код сразу. В случае шаблона компилятор сначала «пишет» для вас реальный класс, заменяя параметры шаблона указанными вами типами. Теперь компилятор компилирует файлы cpp один за другим. Он не знает о других файлах cpp и ничего не может использовать из других файлов cpp. Допустим, ваша реализация Tree: add выглядит следующим образом:

void Tree::add(T& newData)
{
    newData.destroyEverything();
}

Это полностью законно, если у вашего T есть метод destroyEverything. Когда компилятор компилирует Class.cpp, он хочет быть уверен, что вы ничего не делаете с T, чего он не знает. Например, Tree<int> не будет работать, потому что int не имеет destroyEverything. Компилятор попытается написать ваш код с использованием int вместо T и обнаружит, что код не компилируется. Но поскольку компилятор «видит» только текущий cpp и все, что он включает, он не сможет проверить функцию add, поскольку он находится в отдельном cpp.

Не будет никаких проблем с

void Tree::add(int& newData)
{
    newData.destroyEverything();
}

реализован в отдельном cpp, потому что компилятор знает, что int является единственным приемлемым типом, и может "рассчитывать на себя", что, когда он доберется до компиляции Tree.cpp, он найдет ошибку.

0 голосов
/ 07 мая 2009

Вы написали в комментарии:

Я рассмотрел это, но функция является частью Tree.h, и я включаю ее. Определена функция: шаблон void Tree :: add (T & newData); Мы называем это следующим образом: priceTree.add (* newPriceNode); тогда как priceTree - это Tree, оба из которых определены в рассматриваемом файле cpp.

вместо:

priceTree.add(*newPriceNode);

попробовать:

priceTree.add(newPriceNode); //no "*" before "newPriceNode"

add () принимает ссылку на узел, а не указатель на узел (согласно вашему определению дерева).

0 голосов
/ 07 мая 2009

Как уже говорили другие, вам нужно показать реализацию Treee :: add () и рассказать нам, как вы ее связываете.

В несвязанной точке, если вы используете такие конструкции, как:

  Band* newBand=NULL;
    try{
        newBand=new Band(bandID,price-priceOffset,votes);
    }
    catch(bad_alloc&){return ALLOCATION_ERROR;}

во всем вашем коде, вы откровенно тратите свое время. Шансы на то, что вы достигнете исчерпания памяти в современной ОС, малы, а шансы на то, что вы сделаете что-нибудь полезное после того, как это произошло, примерно равны нулю. Вам будет намного лучше просто сказать:

Band * newBand = new Band ( bandID, price - priceOffset, votes );

не возможно:

Band newBand( bandID, price - priceOffset, votes );

и забыть об обработке исключения в этом случае.

0 голосов
/ 07 мая 2009

Вы получаете ошибки компоновки, а не ошибки компилятора. Это говорит нам о том, что компилятор знал, что это за функция Tree::add(), но не имел определения. В Tree.h я вижу объявление функции add (), но не определение. Это выглядит странно для меня; Кто-нибудь знает, откуда появился Tree.h?

Обычно шаблонный класс поставляется с определениями функций-членов во включаемом файле, так как функции должны быть где-то созданы, и самое простое для компилятора - создать экземпляр при использовании и позволить компоновщику разобраться. Если бы определения были в Tree.h, я бы ожидал, что все будет работать как запланировано.

Итак, я собираюсь выйти на конечность и предположить, что определения находятся в отдельном файле, не связаны в нем, и что в других местах есть положения для создания экземпляров для базовых типов, таких как Tree<int>. Это, по-видимому, упрощает компиляцию, так как обычно эти вещи компилируются в нескольких местах, и это требует времени.

В этом случае вам нужно найти, где создается экземпляр Tree<int>, и добавить экземпляр для вашего класса.

Я мог бы быть далеко от базы, но мое объяснение соответствует фактам, которые вы дали.

Редактировать после первых комментариев :

Шаблоны несколько сложнее, чем обычные функции, что обычно не является реальной проблемой. Если бы определения всех вызовов были в Tree.h, то Festival.cpp мог бы создать экземпляр Tree<Band>, и все было бы круто. Это обычная техника, и вы столкнулись с этой проблемой, потому что вы ее не используете.

Когда вы пишете функцию, она компилируется, и компоновщик найдет ее. Любая подпрограмма, вызывающая эту функцию, должна знать прототип функции, поэтому она будет знать, как ее вызвать. Когда вы пишете шаблон, вы не пишете ничего, что попадет непосредственно в программу, но любое использование шаблона считается записью всех функций.

Следовательно, где-то в вашей программе должно быть какое-то использование Tree<Band>, чтобы была скомпилирована функция Tree<Band>::add(). Определение Tree<T>::add должно быть доступно для компилятора, когда создается экземпляр Tree<Band>, потому что в противном случае компилятор не имеет представления, что компилировать. В этом случае он генерирует вызов функции, будучи уверенным, что вы убедитесь, что функция скомпилирована в другом месте.

Следовательно, вам нужно создать экземпляр Tree<Band> внутри файла, который имеет доступ к обоим определениям Tree<T> и Band. Это, вероятно, означает, что файл, который включает или включает Tree.cpp и включает Festival.h.

Компоновщик уже использует Tree.cpp, но в Tree.cpp не определено Tree<Band>, поэтому для компоновщика это бессмысленно. Шаблоны полезны только для компилятора, а компоновщик работает только с тем, что компилятор сгенерировал из шаблонов.

Самый быстрый способ решить эту проблему - взять определения из Tree.cpp и поместить их в Tree.h. К сожалению, это может увеличить время компиляции и ссылки. Другой метод заключается в создании экземпляров всех шаблонов, используемых в Tree.cpp, чтобы они были скомпилированы там.

0 голосов
/ 07 мая 2009

Обновление: использование функций дерева с примитивным типом не приводит к ошибке компоновки. Я обновил свой вопрос в свете некоторых вещей, которые были сказаны.

0 голосов
/ 07 мая 2009

Вы уверены, что try / catch имеет какое-либо отношение к этому? Что произойдет, если вы просто закомментируете строки try и catch, оставите остальной код без изменений и создадите его?

Возможно, вам просто не хватает библиотеки, которая определяет Tree::add(class Price &) в вашей строке ссылки.

...