当前位置: 首页 > 知识库问答 >
问题:

基于文本的博弈非分支if语句

鲍建业
2023-03-14

我正在用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语句的情况下实现这个概念?

共有3个答案

李振国
2023-03-14

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

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? */
}
容磊
2023-03-14

看到这个问题和我的回答。

简短版本:

为每个“分支”使用单独的函数或对象。使用< code>std::map代替“if/else”。将用户输入id映射到处理它们的函数。

宰父淳
2023-03-14

您可以通过使用数据驱动的设计来摆脱分支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!";
}
 类似资料:
  • 我试图测试自己理解递归的能力,所以我给自己一个任务,在递归中做跳跃游戏练习 给定一个非负整数数组,您最初位于数组的第一个索引处。数组中的每个元素代表该位置的最大跳转长度。你的目标是在最小的跳跃次数内达到最后一个指数。 https://leetcode.com/problems/jump-game-ii/ 我试图修改这部分代码,但它没有出现在调试器上,因此我没有真正看到这部分代码中的问题 如果有人能

  • if单分支 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> int main01(void) { if (-2 + 2)//只分辨0与非0的情况 { system("msconfig"); } } int main02(void) { if (0)

  • 我已经通读了所有其他文章,我没有找到我的答案。 我从一个带有对象选项的JOptionPane开始。当用户选择动物或栖息地时,会出现另一个带有选项的下拉列表。现在我正在努力让另一个信息弹出窗口在狮子被选中时打开。但我不能让它工作。 我还查看了其他人的输入,以便能够显示这些下拉列表。但他们使用的是JFrame,因此在NetBeans之外会打开另一个java窗口。 我现在要做的是为下拉列表中的选项输入i

  • 我正在制作一个2人迷宫跑者游戏,我遇到了一些键盘事件的麻烦。如果两个玩家同时击中一个键,只有玩家1移动,因为我的代码首先测试的是Player1的事件。在python和pygame中是否有一种方法可以同时检查这两个事件?下面是我的player1课的一部分: 很抱歉出现了代码块,但这对您理解它是如何工作的都是必要的。我为Player2提供了一个几乎相同的类,但使用了不同的键盘控制(WASD而不是箭头键

  • 请看下面我附加的代码,然后我的问题在最后。 背景:我正在学习java并试图理解一个概念。使用这个程序,我试图创建一个带有按钮的小gui,当单击该按钮时,该按钮将为JLabel分配字符串值“the new label”。但是,如果单击第二次,我想使用相同的按钮将标签更改为“Newerer label”,如果单击第三次,我想使用相同的按钮将标签更改为“Newerer label”。我试图使用一个if/

  • 在每次移动中,Nick可以从堆栈A或B堆栈的顶部移除一个整数。 Nick保存从两个堆栈中移除的整数的运行和。 如果Nick在任何一点上的运行和大于游戏开始时给定的某个整数X,他将被取消比赛资格。 我不明白算法,它看起来像DP,对我来说,有人能帮助我的方法/算法吗? **