0

我有一组用 makefile 编译的文件来制作一个单独的链接散列程序。该程序一直运行,直到我添加了插入、删除和包含函数的代码。我直接从书中提取了代码,但是我遇到了一个我无法弄清楚的模棱两可的错误,希望这里有人可以帮助识别它。我没有发布整个程序,因为我有根据地猜测错误的原因不会在这段代码之外找到(但我可能是错的)

有问题的错误是:

Undefined                       first referenced
 symbol                             in file
hash(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)hashApp.o

此外,不确定这是否相关,但如果我尝试使用函数本身编译 .cpp 文件,我会得到:

Undefined                       first referenced
 symbol                             in file
main                                /opt/csw/gcc3/lib/gcc/sparc-sun-solaris2.8/3.4.6/crt1.o
ld: fatal: Symbol referencing errors. No output written to a.out
collect2: ld returned 1 exit status

以下是函数,字符串在列表向量中进行散列:

template <class HashObj>
bool HashTable<HashObj>::contains(HashObj &item)
{
  const list<HashObj> & whichList = theLists[ myhash( item ) ];
  return find( whichList.begin( ), whichList.end( ), item ) != whichList.end( );    
}        

template <class HashObj>
bool HashTable<HashObj>::insert(const HashObj &item)
{
    list<HashObj> & whichList = theLists[ myhash( item ) ];
    if( find( whichList.begin( ), whichList.end( ), item ) != whichList.end( ) )
        return false;
    whichList.push_back( item );
    return true; 
} 

template <class HashObj>
bool HashTable<HashObj>::remove(const HashObj &item)
{
  list<HashObj> & whichList = theLists[ myhash( item ) ];
  typename list<HashObj>::iterator itr = find( whichList.begin( ), whichList.end(), item );

  if( itr == whichList.end( ) )
  return false;

  whichList.erase(itr);
  return true;
 } 

这是同一文件中的 myhash 函数:

 template <class HashObj>
int HashTable<HashObj>::myhash(const HashObj &item) const
{
    int hashVal = hash(item);

    hashVal %= theLists.size();
    if (hashVal < 0)
        hashVal += theLists.size();

    return hashVal;
}

上面的 .cpp 代码包含一个 hashTable.h,它又包含 hashPrototypes.h

在 hashPrototypes.h 中是

int hash(int key);
int hash(const string &key);

我的哈希函数是从一个生成文件编译的,该文件根据你输入的内容创建一个可执行文件。例如,我正在使用 hash1.cpp,因此通过键入 make HASH=hash1,它应该将它们全部编译在一起。

这是我的 hash1.cpp 代码:

#include "hashTable.h"
#include <cmath>
#include <cstdlib>
using namespace std;

template <class HashObj>
int hash(const HashObj &item)
    {
    int hashVal = 0;

    for( int i = 0; i < item.length( ); i++ )
        hashVal = 37 * hashVal + item[ i ];

    return hashVal;
}

如果您认为错误在 makefile 中,这里是 makefile 代码:

#   Make file for hashing
#   Executable for the program will be in: hashTest

#default function is looked for in hashS1
#to give it another function make=filename without the suffix
HASH = hashS1

$(HASH)Test:  $(HASH).o hashTable.o hashApp.o
    g++ -o $(HASH)Test $(HASH).o hashTable.o hashApp.o

hashApp.o:      hashTable.h hashPrototypes.h hashApp.cpp hashTable.cpp
    g++ -c hashApp.cpp

hashTable.o:    hashTable.h hashTable.cpp $(HASH).cpp
g++ -c hashTable.cpp

$(HASH).o:  hashPrototypes.h $(HASH).cpp
g++ -c $(HASH).cpp

clean:
rm -f *.o 
touch *
4

2 回答 2

3

问题是您已将模板代码放入 cpp 文件中。所有模板代码都应该放在头文件中。否则,在使用这些模板时会出现链接错误。

于 2013-10-08T04:55:06.490 回答
0

发现了这个问题,它比我想象的要简单,但约翰的回答是关于模板代码在哪里有帮助。

原来我需要让我的哈希函数(不是 myhash)成为一个非模板类,并让它接受一个字符串变量。在查看原型后,这对我来说很明显:

int hash(const string &key);

在我最初创建定义时,我只是假设它将是一个模板类,就像其他所有东西一样!

于 2013-10-08T14:14:32.067 回答