开发者

C++ Includes in a cycle

Please consider the following three simplified files:

student.h:

#ifndef STUDENT_H
#define STUDENT_H

#include "course.h"

class Student
{
private:
    Course someCourse;
};

#endif

course.h:

#ifndef COURSE_H
#define COURSE_H

#include "student.h"

class Course
{
private:
    Student someStudent;
};

#endif

and main.cpp:

#include "student.h"
int main();

This wouldn't compile giving me

error C2146: syntax error : missing ';' before identifier 'someStudent'

It would produce a lot more errors (even for the correct portions of code) in a more complicated program. I guess the design is wrong: Student includes Course and Course includes Student. What I want to represent with it is that a st开发者_运维知识库udent takes several courses and a course has has several students (I use vectors in a full program, avoided them here for simplicity). Any advice how this would be possible?

Thanks in advance, Vlad.

UPDATE: Thanks for fast replies. Forward declaration of Student class in Course class (and removing #include "student.h") seems to do the job. Sorry, I thought it wouldn't matter here, but in fact I am using vectors of const pointers in each of them (since a Student shouldn't be able to control a Course and a Course shouldn't be able to control a Student), as:

vector<const Student* const> students; // in Course class


This is going circular, as long as you declare someCourse and someStudent as non-pointer members of class Student and Course respectively (as you've done), because the compiler sees the definition of Student, it needs to know its size which in turn means, it needs to know the size of its all members, including Course which is one of them. But to know the size of Course, it needs to know the size of Student. That becomes circular.

So you need to break this circle, by declaring at least one of them as pointer. For example, you can do this:

#ifndef STUDENT_H
#define STUDENT_H

//#include "course.h" //commenting out, as it is not needed!

class Course; //this is called forward declaration

class Student
{
private:
    Course *pSomeCourse; //pointer
};

#endif

Also note that when you declare pSomeCourse as pointer of type Course*, you don't need to include the header file in which Course is defined. Just forward declaration of the class Course is enough, as I did in the above code.

The reason why it works because the size of pointer of any class is same, and the compiler need not know the size of the class, in order to know the size of the pointer of the same class. In other words, the compile can know sizeof(Course*) without even knowing sizeof(Course).


In addition to Nawaz answer:

if you want to access members of Course from members of Student, you need to include Course.h in the .cpp - File where you define the methods of Student.

With g++ you'll get an error like "invalid use of incomplete type" otherwise.


If you want to link two classes you will have to use forward declaration and a pointer to an instance of the declared type in one of the class interfaces. Another interface can stay the same as long as it includes declaration of the member variable's type interface.

course.h:

#ifndef COURSE_H
#define COURSE_H


class Student;
class Course
{
private:
    Student* someStudent;
};

#endif

student.h:

#ifndef STUDENT_H
#define STUDENT_H

#include "course.h"

class Student
{
private:
    Course someCourse;
};

#endif


You can't do that, you have to convert a least one of them to a pointer.

0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新问答

问答排行榜