C++ пытается использовать операторы #ifndef и #include

Итак, я профессионал HTML/Javascript/PHP, но я пытаюсь изучать C++. Я все еще новичок в этом, и у меня есть проект программирования C++, с которым у меня есть ошибки.

Файл, который содержит int main() - это football.cpp, и у меня есть файлы.h и.cpp для трех классов, которые мне нужно использовать: Game, Team и Date. Каждый экземпляр команды содержит вектор игр, а каждая игра содержит дату. Дата не содержит экземпляров какого-либо другого класса.

Я пытаюсь найти способ использовать операторы #include и #ifndef в верхних частях файлов, чтобы не получать ошибок при компиляции, но я не нашел комбинацию, которая работает. Я не уверен, есть ли другие ошибки. Вот мои текущие разделы #include, не считая других библиотек:

football.cpp

#include "game.h"
#include "team.h"
#include "date.h"

team.h

#ifndef __game_h_
#define __game_h_
#endif

team.cpp

#include "team.h"
#ifndef __game_h_
#define __game_h_
#endif

game.h

#ifndef __date_h_
#define __date_h_
#endif

game.cpp

#include "game.h"
#ifndef __date_h_
#define __date_h_
#endif

date.cpp

#include "date.h"

Я использую Cygwin G ++ компилятор, и строка, которую я использую для его компиляции:

g++ football.cpp team.cpp game.cpp date.cpp -o football.exe

Вот все ошибки, которые я получаю: (ПРЕДУПРЕЖДЕНИЕ, СТЕНА ТЕКСТА)

$ g++ football.cpp team.cpp game.cpp date.cpp -o football.exe

In file included from football.cpp:9:0:
game.h:15:96: error: ‘Date’ has not been declared
game.h:24:1: error: ‘Date’ does not name a type
game.h:37:1: error: ‘Date’ does not name a type
football.cpp: In function ‘int main(int, char*)’:
football.cpp:65:70: error: no matching function for call to ‘Game::Game(std::string&, std::string&, int [5], int [5], Date&)’
game.h:15:1: note: candidates are: Game::Game(std::string, std::string, const int, const int*, int)
game.h:14:1: note: Game::Game()
game.h:10:12: note: Game::Game(const Game&)
In file included from team.cpp:4:0:
team.h:24:8: error: ‘Game’ was not declared in this scope
team.h:24:12: error: template argument 1 is invalid
team.h:24:12: error: template argument 2 is invalid
team.cpp: In member function ‘float Team::getStat3()’:
team.cpp:36:26: error: request for member ‘size’ in ‘((Team*)this)->Team::games’, which is of non-class type ‘int’
team.cpp:37:21: error: invalid types ‘int[int]’ for array subscript
team.cpp:37:50: error: invalid types ‘int[int]’ for array subscript
team.cpp:38:21: error: invalid types ‘int[int]’ for array subscript
team.cpp:38:47: error: invalid types ‘int[int]’ for array subscript
team.cpp:38:76: error: invalid types ‘int[int]’ for array subscript
team.cpp:38:106: error: invalid types ‘int[int]’ for array subscript
team.cpp: In function ‘bool compare2(Team, Team)’:
team.cpp:45:39: error: request for member ‘size’ in ‘t1.Team::games’, which is of non-class type ‘const int’
team.cpp:46:39: error: request for member ‘size’ in ‘t2.Team::games’, which is of non-class type ‘const int’
team.cpp:50:17: error: request for member ‘size’ in ‘t1.Team::games’, which is of non-class type ‘const int’
team.cpp:50:35: error: request for member ‘size’ in ‘t2.Team::games’, which is of non-class type ‘const int’
team.cpp:52:24: error: request for member ‘size’ in ‘t1.Team::games’, which is of non-class type ‘const int’
team.cpp:52:43: error: request for member ‘size’ in ‘t2.Team::games’, which is of non-class type ‘const int’
team.cpp: In function ‘bool compare3(Team, Team)’:
team.cpp:62:29: error: passing ‘const Team’ as ‘this’ argument of ‘float Team::getStat3()’ discards qualifiers
team.cpp:63:29: error: passing ‘const Team’ as ‘this’ argument of ‘float Team::getStat3()’ discards qualifiers
In file included from game.cpp:5:0:
game.h:15:96: error: ‘Date’ has not been declared
game.h:24:1: error: ‘Date’ does not name a type
game.h:37:1: error: ‘Date’ does not name a type
game.cpp: In constructor ‘Game::Game()’:
game.cpp:26:3: error: ‘date’ was not declared in this scope
game.cpp:26:15: error: ‘Date’ was not declared in this scope
game.cpp: At global scope:
game.cpp:29:94: error: ‘Date’ has not been declared
game.cpp:29:1: error: prototype for ‘Game::Game(std::string, std::string, int*, int*, int)’ does not match any in class ‘Game’
game.h:10:12: error: candidates are: Game::Game(const Game&)
game.h:15:1: error: Game::Game(std::string, std::string, const int*, const int*, int)
game.cpp:13:1: error: Game::Game()
game.cpp: In member function ‘int Game::getVisitingScore(int) const’:
game.cpp:80:10: error: ‘visitingScores’ was not declared in this scope
game.cpp: At global scope:
game.cpp:94:1: error: ‘Date’ does not name a type


Отредактирую, если понадобятся дальнейшие разъяснения, хотя я не думаю, что это будет.

Любая помощь будет высоко ценится.

4 ответа

Решение

Я думаю, что вы неправильно понимаете, как использовать охранников.

#ifndef __game_h_
#define __game_h_
// ...
#endif

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

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

Я предполагаю, что ваше неправильное использование включенных охранников фактически предотвращает определение некоторых типов вообще.

Как уже упоминалось, не используйте имена, начинающиеся с двойного подчеркивания.

Например, добавьте include guard в заголовок вашей команды, чтобы предотвратить многократное включение этого файла, и дайте include guards имя, которое отражает модуль, который они защищают:

// team.h
#ifndef TEAM_H
#define TEAM_H

// ... code for team.h in here

#endif    // TEAM_H

Похоже, у вас есть две проблемы: во-первых, вы используете include guard в ваших исходных файлах; во-вторых, неправильный порядок, в котором вы включили ваши файлы.

Сначала включите охрану. Другие уже ответили на этот вопрос, поэтому я буду краток. Защитные заглушки используются только в заголовочных файлах, чтобы классы / типы не объявлялись более одного раза. Например, если бы я имел:

"game.h":

class Game {};

"game.cpp":

#include "game.h"
#include "game.h"  // ERROR:  class Game is declared twice.

Чтобы исправить это, я бы использовал include guard в "game.h":

#ifndef GAME_H_INCLUDED
#define GAME_H_INCLUDED

class Game {};

#endif

При первом включении файла GAME_H_INCLUDED не определяется, поэтому Game объявляется. Во второй раз, когда включается include, определяется GAME_H_INCLUDED, поэтому объявление пропускается.

Проблема, с которой вы столкнулись, заключается в том, что ваши включенные средства защиты в ваших исходных файлах будут пропускать всю реализацию:

сломанный "game.cpp":

#include "game.h"
#ifndef GAME_H_INCLUDED  //  This is defined in "game.h" so everything will be
                         //  skipped until #endif is encountered.
#define GAME_H_INCLUDED

//  Implementation of Game class

#endif

Во-вторых, неправильный порядок включения заголовков. Я предполагаю, что у вас есть что-то вроде этого:

"game.h"

#ifndef GAME
#define GAME

class Game
{
    Date mDate;  // Member of type Date, declared in "date.h"
};

#endif

"date.h"

#ifndef GAME
#define GAME

class Date
{
};

#endif

"game.cpp"

#include "game.h"  // ERROR:  Game relies on the Date class,
                   //         which isn't included yet
#include "date.h"

Чтобы это исправить, поменяйте местами порядок включений в "game.cpp":

"game.cpp"

#include "date.h"
#include "game.h"  // Fine, we know what a Date is now.

Или включите "date.h" в "game.h":

"game.h"

#include "date.h"
#ifndef GAME
#define GAME

class Game
{
    Date mDate;  // Member of type Date, declared in "date.h"
};

#endif

"game.cpp"

#include "game.h"  // Still fine, "date.h" is included by "game.h"

Вам не нужны охранники в ваших файлах C++

Ошибка в том, что вы должны использовать include guard только в ваших заголовочных файлах

Если у тебя есть

BaseClass.h
struct HumanEntity { }

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

#ifndef __MYGUARD
#define __MYGUARD

... // here is your header code

#endif
Другие вопросы по тегам