0

我有 3 个类,每个类都与其他类相关,其中 3 个使用相同的函数,所以我决定创建 functions.h 文件,我将放置每个人都使用的“外部”函数。
所以对于演示:

functions.h
 len function
 subs function
A.cpp
  #include "functions.h"
  #include "A.h"
  #include "B.h"
  cout<<len(word);
B.cpp
   #include "functions.h"
   #include "B.h"
   #include "A.h"
   cout<<subs(word,0,1);
C.cpp
   cout<<len(word);

所有这些都包含具有特定功能的functions.h文件,对于我正在使用标头保护的头文件。
当我构建项目时,我得到了错误:

Error   1   error LNK2005: "int __cdecl len(char *)" (?len@@YAHPAD@Z) already defined in A.obj  C:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\1033\Ass3\Ass3\B.obj Ass3

同样的错误是 subs(function)。
有什么建议为什么会发生?
谢谢!


编辑:
functions.h

#ifndef FUNCTIONS_H
#define FUNCTIONS_H
#include <math.h>

  int len(char *w){
     int count=0;
    int i=0;
  while (w[i]!='\0')
      {
          count++;
          i++;
      }
 //cout<<"Length of word is:"<<count<<"\n";
  return count;

}
  char *subs(char *w,int s,int e){
    int i,j=0;
    int size=0;

     size=abs((e))+1;
     //cout<<"new size is:"<<size<<"\n";
     char *newW=new char[size];


     for(i = 0 ;i<e; i++)
        {

           newW[i]=w[s];
           s++;

        }

    newW[i]='\0';
    return newW;

   }

   #endif

单词.cpp

#include "Word.h"
#include "functions.h"

Word::Word(char *_word){word=_word;}
bool Word::equals(char* _word){
    cout<<"len of the first word is: "<<len(word)<<" and len of the checked word is: "<<len(_word)<<endl;
    if(len(word)!=len(_word))
        return false;
    else{
    for(int i=0;i<len(word);i++)
        if(word[i]!=_word[i])
            return false;
    }
    return true;

}
char Word::getWord(){
    char *nW = new char[len(word)];
    //cout<<"len of word is:"<<len(word);
    int l=len(word);
    for(int i=0;i<l;i++)
        {
            nW[i]=word[i];
            cout<<nW[i];
    }

    return *nW;

}
void Word::print(char *word){cout<<word;}
void Word::print(){cout<<word<<" ";}

句子.cpp

#include "Sentence.h"
#include "Word.h"
#include "functions.h"
Sentence::Sentence()
{

    char* sentence=new char[300];
    cout<<"Entere sentence"<<endl;
    cin.getline(sentence,300);
    int i,j=0,lastIndex=0,count=0;


    int l=len(sentence);
    cout<<"Size of sentence is: "<<l<<"\n";

    for(i=0;i<l;i++){
        //' ' ,'.', ',', '?', ':', '!' ,'\N','\r' ',', '\t', ',', '-'
        if (sentence[i]==' '||
            sentence[i]=='.'||
            sentence[i]==','||
            sentence[i]=='?'||
            sentence[i]==':'||
            sentence[i]=='!'||
            sentence[i]==';'||

            sentence[i]=='-'){
            count++;
            if(count==1)
                {
                    //cout<<subs(sentence,0,i);
                //cout<<"Start Index: 0 and Length is: "<<i<<"\n";
                words[j]=new Word(subs(sentence,0,i));



                    lastIndex=i;

                    j++;
                }
            else{


        //cout<<subs(sentence,lastIndex+1,i-lastIndex-1);
                //cout<<"Start Index: "<<lastIndex+1<<" and Length is: "<<i-lastIndex-1<<"\n";
            words[j]=new Word(subs(sentence,lastIndex+1,i-lastIndex-1));

            lastIndex=i;
            j++;
            }
        }

    }


    if(lastIndex==0){
        //cout<<subs(sentence,0,l);

        words[j]= new Word(subs(sentence,0,l));
    }
    else{
        //cout<<subs(sentence,lastIndex+1,i);
        //cout<<"Start Index: "<<lastIndex+1<<" and length is: "<<i-lastIndex-1<<"\n";
        words[j]= new Word(subs(sentence,lastIndex+1,i-lastIndex-1));

    }
    wordNum=count+1;

}
bool Sentence::containsWord(char* _word){

    for(int i=0;i<200;i++){


        if(words[i]->equals(_word))
                return true;

    }
    return false;

}

int Sentence::getWordNum(char *_word){

    for(int i=0;i<200;i++){
        cout<<words[i]->getWord();
        if(words[i]->equals(_word))
                return i+1;

    }
    return -1;

}
int Sentence::getWords(){return wordNum;}
int Word::getLen(){

    return len(word);
}
4

2 回答 2

4

似乎您已将函数实现放入functions.h. 将原型留在那里并将实现移动到单独的文件中,例如functions.cpp.

这应该可以解决问题。

更新您遇到的错误的更多解释。

当您使用#include它时,它将在编译前由处理器处理(将其视为将文件内联.h到源文件中)。这个过程发生在每个编译单元,即.cpp文件(把它想象成树的根)。

现在,由于包含树的“根”是不同的(A.cpp, B.cpp, ...),因此包含保护不会帮助您避免函数实现被内联到每个编译单元中。

这将导致链接错误,因为这些函数的实现不止一种。

于 2013-07-16T08:17:54.447 回答
2

每个编译单元(即 A.cpp、B.cpp 和 C.cpp)都有一个包含在 .cpp 中的每个函数的定义functions.h

包含警卫对此没有帮助,因为它们仅在编译单元边界内产生影响(即,如果您直接或间接地functions.h在 A.cpp 中包含两次)

可能的解决方案是:

  1. 您在 中声明函数functions.h,并在单独的编译单元中定义它们,例如functions.cpp(这是您示例中的方法)
  2. 你在模板函数中创建每个functions.h函数(只有在有意义的情况下才这样做)
  3. 您将每个函数声明functions.h为静态(这只是一种解决方法,我绝不建议您这样做!!!)

PS关于第3点。这可能是有道理的,所以你可能想知道是static什么。当引用自由函数时,static关键字将此函数标记为编译单元的本地函数。因此,该函数将不会在链接表中导出,并且无法从另一个编译单元中调用。这就是“技巧”起作用的原因,尽管您仍然有多个定义,它们都是每个编译单元的“私有”,并且没有可能的冲突。当您使用不属于“接口”并且仅作为本地辅助函数的自由函数时,这特别有用。

于 2013-07-16T08:23:13.657 回答