I have a namespace defined in one header file and used in another, but it cannot be found. Specifically, a namespace called “players” defined in “players/Players.hpp” and used in a file called “players/Ownable.hpp” cannot be found in a file called “combat/Targetable.hpp”
The errors are
...\source\combat\Targetable.hpp(7): 'players' : is not a class or namespace name
...\source\combat\Targetable.hpp(7): 'Ownable' : base class undefined
Obviously it’s some syntax thing I don’t understand. I’ve spent some time simplifying the code so it looks silly, but bear with me.
// source/players/Players.hpp:
#ifndef PLAYERS_HPP
#define PLAYERS_HPP
#include "../Headers.hpp"
namespace players {
class Player{
// this class compiles fine.
// There used to be a "Players.cpp" but it's been simplified away
public:
int getID(){ return 0; }
int getTeam(){ return 0; }
string getName(){ return ""; }
Vec3 getColor(){ return Vec3(0.0,0.0,0.0); }
};
}
#endif
And players/Ownable.hpp, which is in the same folder as Player.hpp and also compiles fine:
// source/players/Ownable.hpp:
#ifndef OWNABLE_HPP
#define OWNABLE_HPP
#include "Players.hpp"
namespace players {
class Ownable;
typedef boost::shared_ptr<Ownable> OwnablePTR;
typedef boost::weak_ptr<Ownable> OwnableWPTR;
class Ownable {
public:
Ownable(){}
Ownable(int playerID) : playerID(playerID){}
bool isAlliedWith(OwnablePTR other){ return false; }
private:
int playerID;
};
}
#endif
Here’s where the fun starts. I have a file at “source/combat/Targetable.hpp”, which is in a different directory than the other two. However, the file itself seems to include fine:
// source/combat/Targetable.hpp:
#ifndef TARGETABLE_HPP
#define TARGETABLE_HPP
#include "../players/Ownable.hpp"
namespace combat{
class Targetable : public players::Ownable { // ERROR
public:
Targetable(int playerID){}
//Targetable(players::Player player);
virtual Vec2 getPosition(){
return Vec2();
}
virtual Vec2 getVelocity(){
return Vec2();
}
};
}
#endif
I’m really hoping this is some silly syntax thing that I’m missing. I’ve even tried
using players::Ownable;
but that A) pollutes the files that include this one, and B) doesn’t fix anything. Any help?
EDIT: GManNickG got it, it was a circular include in the Headers.hpp file. Thanks!
You have a circular include.
First consider the purpose of include guards:
The inclusion of
c.hppwill end up includea.hpptwice. The first time, the guards are not defined and everything is okay, and the second time the guards prevent redefinitions. This is what we want.This does not work when you have a loop, though. (It will prevent it, which is good, but it does “too well” because the guard is defined right after its tested, which means the contents of the header haven’t actually yet been processed). Consider this instead:
Which is similar to what you have.
x.cppincludesc.hpp, which is the first time it’s been included so it definesC_HPP. Thenc.hppincludesb.hpp, which includesa.hpp. Thena.hppincludesc.hppand finds thatC_HPPhas already been defined, so the include does nothing.Assuming
a.hppstill manages to compile (that is,c.hppisn’t actually needed), thena.hppfinishes, thenb.hppfinishes, thenc.hppfinally actually defines its contents before returning tox.cpp.The solution is to minimize the amount of headers you include. Use forward declarations, and most of all: do not use ‘include everything’ headers! These are terrible. And I suspect that’s what
Headers.hppis.