0

我正在定义一个 GameState 类和一个 MainMenuGameState 类。前者是一个抽象类,后者是继承它。但不知何故,我无法覆盖它的属性。

游戏状态.h

#ifndef _GAME_STATE_H_
    #define _GAME_STATE_H_

#include <SDL2/SDL.h>

class GameState {
    public:
        virtual void loop(Uint32 deltaTime) = 0;
        virtual void render() = 0;
        virtual void event(SDL_Event * event) = 0;

        bool stopRenderPropagation = false;
        bool stopLoopPropagation = false;
};

#endif

MainMenuGameState.h

#ifndef _MAIN_MENU_GAME_STATE_H_
    #define _MAIN_MENU_GAME_STATE_H_

#include "../Game.h"

class MainMenuGameState : public GameState {
    public:
        MainMenuGameState(Game * pGame);

        void loop(Uint32 deltaTime);
        void render();
        void event(SDL_Event * event);

        bool stopRenderPropagation = true;
        bool stopLoopPropagation = true;

    private:
        Game * game;

        int xOffset = 0;
        int yOffset = 0;
};

#endif

因此,在实例化 MainMenuGameState 对象后,我期望stopRenderPropagationstopLoopPropagationtrue,但它们是false

由于某种原因,我也没有运气在构造函数中覆盖它们。

MainMenuGameState::MainMenuGameState(Game * pGame) {
    game = pGame;

    xOffset = rand() % 20;
    yOffset = rand() % 20;

    stopRenderPropagation = true;
    stopLoopPropagation = true;
}

在那之后,它们仍然是真实的。我不知道这是我的构造函数的问题,或者我是否误解了 C++ 中的多态性。

MainMenuGameState 的实例存储在 avector<GameState *>中,这可能是问题吗?我正在访问这样的属性:

if(gameStates.begin() != gameStates.end()) {
    std::vector<GameState *>::iterator it = gameStates.end();
    do {
        --it;
    } while(it != gameStates.begin() && (*it)->stopLoopPropagation == false);

    while(it != gameStates.end()) {
        (*it)->loop(deltaTime);
        ++it;
    }
}

谢谢您的帮助!

4

3 回答 3

6

您的派生类声明了另外几个与基类中的成员同名的成员,因此“隐藏”了基类。

您应该在构造函数中接受这些成员的初始值,或者如果它们是永远不会改变的类的固定属性,您应该将它们设为成员函数,而不是像

class GameState {
    public:
        ...    
        virtual bool stopRenderPropagation() { return false; }
        virtual bool stopLoopPropagation() { return false; }
};

class MainMenuGameState : public GameState {
    public:
        ...    
        bool stopRenderPropagation() { return true; }
        bool stopLoopPropagation() { return true; }
        ...
};
于 2013-09-24T06:54:48.913 回答
2

继承数据成员(您的布尔值)与继承和重载方法的工作方式不同。尝试将布尔值作为受保护的继承数据成员(未初始化),并在相应的子类构造函数中初始化它们。

class MainMenuGameState : public GameState {
    public:
        MainMenuGameState(Game * pGame);

        void loop(Uint32 deltaTime);
        void render();
        void event(SDL_Event * event);

    protected:
        bool stopRenderPropagation;
        bool stopLoopPropagation;

    private:
        Game * game;

        int xOffset = 0;
        int yOffset = 0;
};
于 2013-09-24T06:55:36.857 回答
1

您在派生类中声明新变量,这会导致这些问题。继承非私有变量:

struct A { int x };
struct B : A {}; // has B::x by inheritance

您可以在构造函数中设置它们而无需重新声明:

struct A { int x; };
struct B : A { B() : x(1) {} };

请注意,声明公共变量通常被认为是不好的做法,更常见的是实现 getter 和 setter:

struct A
{
    int x() const { return x_; }
    int & x() { return x_; }

    private:
        int x_;
};
于 2013-09-24T06:59:04.170 回答