用C++编写一个2048的小游戏

news/2025/1/30 13:42:07/

以下是一个简单的2048游戏的实现。这个实现使用了控制台输入和输出,适合在终端或命令行环境中运行。


2048游戏的实现


1.游戏逻辑
2048游戏的核心逻辑包括:

• 初始化一个4x4的网格。

• 随机生成2或4。

• 处理玩家的移动操作(上、下、左、右)。

• 合并相同的数字。

• 检查游戏是否结束(胜利或失败)。


2.代码实现


```cpp
#include <iostream>
#include <vector>
#include <cstdlib>
#include <ctime>
#include <algorithm>

using namespace std;

const int GRID_SIZE = 4;
const int WINNING_SCORE = 2048;

void initializeGrid(vector<vector<int>>& grid) {
    for (int i = 0; i < GRID_SIZE; ++i) {
        grid.push_back(vector<int>(GRID_SIZE, 0));
    }
}

void printGrid(const vector<vector<int>>& grid) {
    for (const auto& row : grid) {
        for (int val : row) {
            cout << val << "\t";
        }
        cout << endl;
    }
    cout << endl;
}

void addRandomTile(vector<vector<int>>& grid) {
    vector<pair<int, int>> emptyCells;
    for (int i = 0; i < GRID_SIZE; ++i) {
        for (int j = 0; j < GRID_SIZE; ++j) {
            if (grid[i][j] == 0) {
                emptyCells.push_back({i, j});
            }
        }
    }
    if (!emptyCells.empty()) {
        pair<int, int> cell = emptyCells[rand() % emptyCells.size()];
        grid[cell.first][cell.second] = (rand() % 2) ? 2 : 4;
    }
}

bool canMove(const vector<vector<int>>& grid) {
    for (int i = 0; i < GRID_SIZE; ++i) {
        for (int j = 0; j < GRID_SIZE; ++j) {
            if (grid[i][j] == 0) return true;
            if (i > 0 && grid[i][j] == grid[i - 1][j]) return true;
            if (j > 0 && grid[i][j] == grid[i][j - 1]) return true;
            if (i < GRID_SIZE - 1 && grid[i][j] == grid[i + 1][j]) return true;
            if (j < GRID_SIZE - 1 && grid[i][j] == grid[i][j + 1]) return true;
        }
    }
    return false;
}

void compress(vector<int>& row) {
    vector<int> temp(GRID_SIZE, 0);
    int k = 0;
    for (int i = 0; i < GRID_SIZE; ++i) {
        if (row[i] != 0) {
            temp[k++] = row[i];
        }
    }
    for (int i = 0; i < GRID_SIZE; ++i) {
        row[i] = temp[i];
    }
}

void merge(vector<int>& row) {
    for (int i = 0; i < GRID_SIZE - 1; ++i) {
        if (row[i] == row[i + 1] && row[i] != 0) {
            row[i] *= 2;
            row[i + 1] = 0;
        }
    }
}

void moveLeft(vector<vector<int>>& grid) {
    for (auto& row : grid) {
        compress(row);
        merge(row);
        compress(row);
    }
}

void moveRight(vector<vector<int>>& grid) {
    for (auto& row : grid) {
        reverse(row.begin(), row.end());
        compress(row);
        merge(row);
        compress(row);
        reverse(row.begin(), row.end());
    }
}

void moveUp(vector<vector<int>>& grid) {
    for (int col = 0; col < GRID_SIZE; ++col) {
        vector<int> temp(GRID_SIZE, 0);
        for (int row = 0; row < GRID_SIZE; ++row) {
            temp[row] = grid[row][col];
        }
        compress(temp);
        merge(temp);
        compress(temp);
        for (int row = 0; row < GRID_SIZE; ++row) {
            grid[row][col] = temp[row];
        }
    }
}

void moveDown(vector<vector<int>>& grid) {
    for (int col = 0; col < GRID_SIZE; ++col) {
        vector<int> temp(GRID_SIZE, 0);
        for (int row = 0; row < GRID_SIZE; ++row) {
            temp[row] = grid[row][col];
        }
        reverse(temp.begin(), temp.end());
        compress(temp);
        merge(temp);
        compress(temp);
        reverse(temp.begin(), temp.end());
        for (int row = 0; row < GRID_SIZE; ++row) {
            grid[row][col] = temp[row];
        }
    }
}

bool isGameOver(const vector<vector<int>>& grid) {
    for (const auto& row : grid) {
        for (int val : row) {
            if (val == WINNING_SCORE) return true;
        }
    }
    return !canMove(grid);
}

int main() {
    srand(static_cast<unsigned int>(time(0)));
    vector<vector<int>> grid;
    initializeGrid(grid);
    addRandomTile(grid);
    addRandomTile(grid);

    while (true) {
        printGrid(grid);
        char move;
        cout << "Enter move (W/A/S/D): ";
        cin >> move;

        switch (move) {
            case 'W':
            case 'w':
                moveUp(grid);
                break;
            case 'A':
            case 'a':
                moveLeft(grid);
                break;
            case 'S':
            case 's':
                moveDown(grid);
                break;
            case 'D':
            case 'd':
                moveRight(grid);
                break;
            default:
                cout << "Invalid move. Try again." << endl;
                continue;
        }

        addRandomTile(grid);

        if (isGameOver(grid)) {
            printGrid(grid);
            cout << "Game Over! " << (grid[0][0] == WINNING_SCORE ? "You Win!" : "You Lose!") << endl;
            break;
        }
    }

    return 0;
}
```

3.代码说明

• 初始化网格:`initializeGrid`函数初始化一个4x4的网格,所有值初始化为0。

• 打印网格:`printGrid`函数打印当前网格的状态。

• 添加随机瓷砖:`addRandomTile`函数在网格的空位置随机添加一个2或4。

• 移动操作:`moveLeft`、`moveRight`、`moveUp`、`moveDown`函数分别处理左、右、上、下移动操作。

• 压缩和合并:`compress`和`merge`函数分别处理行或列的压缩和合并操作。

• 检查游戏结束:`isGameOver`函数检查游戏是否结束(胜利或失败)。


4.运行游戏
将上述代码保存为一个C++文件(例如`2048.cpp`),然后使用C++编译器编译并运行它。例如,使用g++编译器:

```sh
g++ -o 2048 2048.cpp
./2048
```

5.游戏玩法

• 使用W/A/S/D键控制方向(上/左/下/右)。

• 游戏目标是合并数字,直到出现2048。

• 如果没有可移动的空位且无法合并,则游戏结束。

希望这个实现对你有帮助!


http://www.ppmy.cn/news/1567573.html

相关文章

2025数学建模美赛|A题成品论文

A题 摘要 本研究提出了一种数学建模方法&#xff0c;旨在帮助考古学家基于台阶磨损模式得出关于台阶使用情况的基本结论。通过模型的建立&#xff0c;我们解决了两个主要任务&#xff1a;首先是从磨损模式中推测台阶的使用频率、行走方向偏好和同时使用人数&#xff1b;其次是在…

LeetCode100之子集(78)--Java

1.问题描述 给你一个整数数组 nums &#xff0c;数组中的元素 互不相同 。返回该数组所有可能的 子集&#xff08;幂集&#xff09;。 解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。 示例1 输入&#xff1a;nums [1,2,3]输出&#xff1a;[[],[1],[2],[1,2],[3],[1…

Web3.0时代的挑战与机遇:以开源2+1链动模式AI智能名片S2B2C商城小程序为例的深度探讨

摘要&#xff1a;Web3.0作为互联网的下一代形态&#xff0c;承载着去中心化、开放性和安全性的重要愿景。然而&#xff0c;其高门槛、用户体验差等问题阻碍了Web3.0的主流化进程。本文旨在深入探讨Web3.0面临的挑战&#xff0c;并提出利用开源21链动模式、AI智能名片及S2B2C商城…

科技快讯 | 2025商业新愿景;豆包大模型1.5 Pro正式发布;ChatGPT每月产生260吨二氧化碳

巨头扎堆入局&#xff0c;人形机器人量产渐行渐近 2025年&#xff0c;人形机器人赛道热度持续升温&#xff0c;工信部《人形机器人创新发展指导意见》指出&#xff0c;2025年人形机器人创新体系初步建立&#xff0c;整机产品实现批量生产&#xff0c;并在多个场景应用。受访人士…

Excel - Binary和Text两种Compare方法

Option Compare statement VBA里可以定义默认使用的compare方法&#xff1a; Set the string comparison method to Binary. Option Compare Binary That is, "AAA" is less than "aaa". Set the string comparison method to Text. Option Compare Tex…

Linux 命令之技巧(Tips for Linux Commands)

Linux 命令之技巧 简介 Linux ‌是一种免费使用和自由传播的类Unix操作系统&#xff0c;其内核由林纳斯本纳第克特托瓦兹&#xff08;Linus Benedict Torvalds&#xff09;于1991年10月5日首次发布。Linux继承了Unix以网络为核心的设计思想&#xff0c;是一个性能稳定的多用户…

【学习笔记】计算机网络(二)

第2章 物理层 文章目录 第2章 物理层2.1物理层的基本概念2.2 数据通信的基础知识2.2.1 数据通信系统的模型2.2.2 有关信道的几个基本概念2.2.3 信道的极限容量 2.3物理层下面的传输媒体2.3.1 导引型传输媒体2.3.2 非导引型传输媒体 2.4 信道复用技术2.4.1 频分复用、时分复用和…

react-bn-面试

1.主要内容 工作台待办 实现思路&#xff1a; 1&#xff0c;待办list由后端返回&#xff0c;固定需要的字段有id(查详细)、type(本条待办的类型)&#xff0c;还可能需要时间&#xff0c;状态等 2&#xff0c;一个集中处理待办中转路由页&#xff0c;所有待办都跳转到这个页面…