基于文本的游戏非分支if语句

时间:2013-07-15 02:16:30

标签: c++ if-statement

我正在使用C ++开发基于文本的游戏。我想知道如果不嵌套这么多if语句,你将如何进入不同的事件。我正在谈论的一个例子是......

if ( userChoice == 1 )
{
     //do something
     cout << "Pick 1 , 2, or 3" << endl;
     if ( userChoice == 1 )
     {
           cout << "pick 1, 2, 3/ some storry line" << endl;
           if ( userChoice == 3 )
           {
                cout << " storyline...123...." << endl;
                if ( userChoice == 2 )
                {
                       //etc
                }
           }
     }
}

如果不分支if语句,我该怎么做这个概念?

4 个答案:

答案 0 :(得分:5)

您可以使用数据驱动设计摆脱分支if语句。

在基础上,你有一个检查输入的循环,并使用当前上下文评估输入,然后更改上下文。

每个if语句都变成'场景'(或房间或状态,选择对你正在编写的程序有意义的名称)。每个场景都有一个id,一个描述,以及一组结果有效的输入和场景编号。

你有一个代表结束的场景。

你有一个循环:

loop until current scene is the end
 Print current scene description
 Ask for input
 Evaluate input

评估输入根据当前场景的有效输入检查输入,如果有效则将当前场景设置为指定场景。

程序将当前场景初始化为第一个场景,然后开始循环。

因此,对于您的示例(显然不完整,但应该让您了解您需要的数据),您将拥有以下场景:

id: 1
description: "Pick 1 , 2, or 3"
inputs: "1" => 2, "2" =>, "3" =>

id: 2
description: "pick 1, 2, 3/ some storry line"
inputs: "1" =>, "2" =>, "3" => 3

id: 3
description: " storyline...123...."
inputs: "1" =>, "2" =>, "3" =>

通常数据来自文件。

以下是一个示例(尚未编译或调试):

struct Scene
{
    Scene(int id_, int description_)
        : id(id_)
        , description(description_)
    {
    }

    int id;
    std::string description;
    std::map<std::string, int> inputToNextScene;
 };

void main(int, char **) 
{
    std::map<int, Scene> scenes;

    int ids = [1,2,3];
    std::string descriptions = ["first", "second", "third"];
    int nextScenes [3][3] = [ [1, 2, 3], [1, 3, 2], [1, 2, 0]];
    std::string input[3] = ["1", "2", "3"];

        for (int i = 0; i != 3; ++i)
        {
            scenes[ ids[i] ] = Scene(ids[i], descriptions);

            Scene& scene = scenes.at(ids[i]);

            for (int j = 0; j != 3; ++j)
            {
                scene.inputToNextScene[ input[j] ] = nextScenes[i][j];
            }
    }

    int currentScene = 1;

    std::string input;

    while (currentScene != 0)
    {
         Scene& scene = scenes.at(currentScene);

         std::cout << scene.description;
         //Maybe put in a prompt and check currentscene against previous before      printing description
         std::cin >> input;
         if (scene.inputToNextScene.find(input) != scene.inputToNextScene.end())
         {
             currentScene = scene.inputToNextScene.at(input);
         }
    }

    cout << "The end!";
}

答案 1 :(得分:1)

请参阅this questionmy answer

简短版本:

为每个“分支”使用单独的函数或对象。 而不是“if / else”,请使用std::map。将用户输入id映射到处理它们的函数。

答案 2 :(得分:0)

我为每个选项创建一个单独的函数:

void doStoryLine1() {
    cout << "Options: ..." << endl;
    if(userChoice == 1)
        fightDragon();
    else if(userChoice == 2)
        fightOrcs();
    else if(userChoice == 3)
        marryPrincess();
    else
        /* invalid option? */
}

void doStoryLine2() {
    ...
}

void selectStoryLine() {
    cout << "Options: ..." << endl;
    if(userChoice == 1)
        doStoryLine1();
    else if(userChoice == 2)
        doStoryLine2();
    else if(userChoice == 3)
        doStoryLine3();
    else
        /* invalid option? */
}

答案 3 :(得分:-2)

您描述的问题似乎是异步编程中的常见问题(例如Node JS)。

您可以使用控制流程编程来突破这些循环,例如序列/瀑布承诺

注意:我之前没有使用过这个标题,但是它应该与JS中的期货一样:Futures vs. Promises

JS中的代码(您可以在C ++中获得想法)

/** Sequence **/
new Sequence()


.then(function(next, arg1, arg2) {
    // stage 1...
    next(arg1, arg2);
})


.then(function(next) {
    // stage 2...
    next();
});



/** Promise **/
var a = Promise();

function stage1() {
    // capture input 1...

    a.fulfill(input1);

}();

a.when(function(err, data) {
    console.log(data);

});