1

私は Java/Python/etc のバックグラウンドから C++ を初めて使用し、次の学期にクラスを受講する前にオブジェクト指向プログラミングを独学しようとしています。

SFML を使用してアニメーション システムを作成しようとしていますが、クラス変数の 1 つに問題があります。インクリメントした後も0にリセットされ続けます。コードから始めて、何が起こっているのかを把握するために使用しているログ出力を続けます。

解決策: C++ を初めて使用する私は馬鹿で、ゲッター関数で自分のクラスの新しいインスタンスを返しました。[class] func() の代わりに [class]& func()... を使用してこれを解決しましたが、今はリファクタリングを行う必要があります。

コード (ヘッダー):

...

typedef std::vector<Frame> frameVect; // (Frame defined above)
typedef std::vector<double> dubVect;

...

class limbAnim
{
private:
    int limbNum;
    int numFrames;
    int curFrame;
    frameVect frames;

public:
    limbAnim(int limb, int nFrames, frameVect F);
    <getters/setters>
    void incCurFrame();

    dubVect incrementAnimation(dubVect curPos, double curRot);
}

コード (cpp):

... (include vector, ofstream, etc)

std::ofstream AnimLog("log.log")

typedef std::vector<Frame> frameVect; // (Frame defined above)
typedef std::vector<double> dubVect;

...

limbAnim::limbAnim(int limb, int nFrames, frameVect F)
{
    limbNum = limb;
    curFrame = 0;
    numFrames = nFrames;
    frames = F;
}

void limbAnim::incCurFrame()
{
    curFrame=curFrame+1;
    if (curFrame >= numFrames)
    {
        curFrame = 0;
        AnimLog << "Greater than." << std::endl;
    }
}

dubVect limbAnim::incrementAnimation(dubVect curPos, double curRot)
{
    AnimLog << limbNum << ", " << numFrames << std::endl;

    if (numFrames > 0)
    {
        AnimLog << curFrame << std::endl;
        dubVect curStepP = frames[curFrame].getStepPos();
        double curStepR = frames[curFrame].getStepRot();

        curPos[0] = curPos[0] + curStepP[0];
        curPos[1] = curPos[1] + curStepP[1];

        curRot = curRot + curStepR;

        incCurFrame();
        AnimLog << "Incremented: " << curFrame << std::endl;
    }

    dubVect retV = curPos;
    retV.push_back(curRot);
    return retV;
}

したがって、リム 6 と 8 の 2 つのフレームでテストしているため、ログ出力は良好に見えますが、これらのリムの curFrame はインクリメント後に 0 にリセットされるようです。

...
5, 0
6, 2
0
Incremented: 1
7, 0
8, 2
0
Incremented: 1
9, 0
...
5, 0
6, 2
0
Incremented: 1
7, 0
8, 2
0
Incremented: 1
9, 0
...(ad nauseam)

編集:インクリメント関数を呼び出すコード。 (main.cpp)

// (Outside main loop.)
Animation walk_anim(12, "assets/anim/walk.dat");

// (Inside main loop.)
for (int i=0; i<12; i++)
{
    dubVect animDat = walk_anim.getLimbFrame(i).incrementAnimation(limbPos[i], curDegs[i]);
    dubVect newPos = getDVect(animDat[0], animDat[1]);
    double newRot = animDat[2];
    curDegs[i] = newRot;
    if (curDegs[i] >= 360)
        curDegs[i] -=360;
    limbPos[i] = newPos;
}

getLimbFrame

Animation::Animation(int lNum, string fName)
{
    numLimbs = lNum;
    fileName = fName;

    // Fill up limbVect with correct # of empty frames.
    for (int i=0; i<numLimbs; i++)
    {
        frameVect emptyFVect;
        limbAnim LA(i, 0, emptyFVect);
        limbFrames.push_back(LA);
    }

    // Boring .dat parsing, populates the 'frames' var of each limbAnim.
    loadAnim();
}

limbAnim Animation::getLimbFrame(int index)
{
    if (index < numLimbs)
    {
        return limbFrames[index];
    }
}
4

2 に答える 2

1

static キーワードを使用してそのメンバー変数を宣言する必要があると思います。そうすれば、それはクラス変数であると言えます。クラスのすべてのインスタンスで共有されます。このような:

static int curFrame;

次に、クラス外から初期化する必要があります。宣言は初期化とは大きく異なることに注意してください。

あなたはそれについてここここで読むことができます

于 2013-06-27T00:55:49.123 に答える