C++實(shí)現(xiàn)控制臺(tái)隨機(jī)迷宮的示例代碼
我全程使用TCHAR系列函數(shù),親測(cè)可以不改動(dòng)代碼兼容Unicode/ANSI開(kāi)發(fā)環(huán)境,功能正常。大概有100行代碼是來(lái)自網(wǎng)絡(luò)的,我也做了改動(dòng),侵權(quán)請(qǐng)聯(lián)系刪除。
這個(gè)代碼不能算是完美,還是會(huì)有輕微的閃屏現(xiàn)象,懶得再加雙緩存了,大家可以自行修改。這里用的是SetConsoleCursorPosition函數(shù)和cls刷新屏幕。
好了,上代碼!VS2015編譯通過(guò)無(wú)警告。其他版本應(yīng)該也沒(méi)問(wèn)題
// C++ Maze main code // Copyright (c) 2020 szx0427 #include <cstdio> #include <Windows.h> #include <conio.h> #include <tchar.h> #include <ctime> using namespace std; #ifdef _UNICODE #include <io.h> #include <fcntl.h> #define CH_RECT L'■' // a rectangle (wall) #define CH_PLAYER L'○' // a circle (player) #define CH_SPACE L' ' // a space (route) #else #define CH_RECT '#' #define CH_PLAYER 'O' #define CH_SPACE ' ' #endif // _UNICODE #define LENGTH (30 + 2 * 2) #define WALL 0 #define ROUTE 1 #define PLAYER 2 static UINT g_Rank = 0; static SHORT g_lives = 3; static BOOL** g_maze = nullptr; void _Create( __in const int x, __in const int y ); void _Print(void); int _CreateAndPrint(void); inline void _die(void) { --g_lives; for (int n = 1; n <= 2; n++) { _tsystem(_T("color fc")); Sleep(70); _tsystem(_T("color 07")); Sleep(70); } } int _tmain(void) { CONSOLE_CURSOR_INFO cci; GetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cci); cci.bVisible = false; SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cci); #ifdef _UNICODE _setmode(_fileno(stdout), _O_U16TEXT); #endif // _UNICODE srand((UINT)time(NULL)); int k; start: k = _CreateAndPrint(); TCHAR ch; bool bExit = false; bool bWin = false; int x = 2, y = 1; while (!bExit && g_lives >= 0 && !bWin) { ch = _gettch(); switch (ch) { case _T('R'): case _T('r'): _tsystem(_T("cls")); x = 2; y = 1; for (int l = 0; l < LENGTH; l++) { free(g_maze[l]); } free(g_maze); k = _CreateAndPrint(); break; case VK_ESCAPE: bExit = true; break; case TCHAR(0xE0): switch (ch = _gettch()) { case TCHAR(72): if (g_maze[x - 1][y] != WALL) { g_maze[x][y] = ROUTE; --x; g_maze[x][y] = PLAYER; } else { _die(); } break; case TCHAR(80): if (g_maze[x + 1][y] != WALL) { g_maze[x][y] = ROUTE; ++x; g_maze[x][y] = PLAYER; } else { _die(); } break; case TCHAR(75): if (g_maze[x][y - 1] != WALL && !(x == 2 && y == 1)) { g_maze[x][y] = ROUTE; --y; g_maze[x][y] = PLAYER; } else { _die(); } break; case TCHAR(77): if (g_maze[x][y + 1] != WALL) { g_maze[x][y] = ROUTE; ++y; g_maze[x][y] = PLAYER; } else { _die(); } break; default: break; } if (x == k && y == LENGTH - 2) { bWin = true; } _Print(); break; default: break; } } x = 2; y = 1; for (int l = 0; l < LENGTH; l++) { free(g_maze[l]); } free(g_maze); if (g_lives == -1) { _tsystem(_T("cls")); _putts(_T("你撞墻次數(shù)超過(guò)限制,本局游戲失?。?)); _putts(_T("如果要再開(kāi)局,請(qǐng)按下[R]鍵!否則,按下其他鍵以退出!")); ch = _gettch(); if (ch == 'R' || ch == 'r') { goto start; } } else if (bWin) { _tsystem(_T("cls")); _putts(_T("恭喜,你贏了!是否要再來(lái)一局?")); _putts(_T("如果要再開(kāi)局,請(qǐng)按下[R]鍵!否則,按下其他鍵以退出!")); ch = _gettch(); if (ch == 'R' || ch == 'r') { goto start; } } return 0; } void _Create(const int x, const int y) { g_maze[x][y] = ROUTE; int dict[4][2] = { { 1, 0 }, { -1, 0 }, { 0, 1 }, { 0, -1 } }; int r, tmp; for (int i = 0; i < 4; i++) { r = rand() % 4; tmp = dict[0][0]; dict[0][0] = dict[r][0]; dict[r][0] = tmp; tmp = dict[0][1]; dict[0][1] = dict[r][1]; dict[r][1] = tmp; } int dx, dy, range, count; for (int j = 0; j < 4; j++) { dx = x; dy = y; range = 1 + (g_Rank == 0 ? 0 : rand() % g_Rank); while (range > 0) { dx += dict[j][0]; dy += dict[j][1]; if (g_maze[dx][dy] == ROUTE) { break; } count = 0; for (int k = dx - 1; k < dx + 2; k++) { for (int l = dy - 1; l < dy + 2; l++) { if (abs(k - dx) + abs(l - dy) == 1 && g_maze[k][l] == ROUTE) { count++; } } } if (count > 1) { break; } --range; g_maze[dx][dy] = ROUTE; } if (range <= 0) { _Create(dx, dy); } } } int _CreateAndPrint(void) { _tprintf(_T("正在分配內(nèi)存...")); g_maze = (int**)malloc(LENGTH * sizeof(int*)); for (int i = 0; i < LENGTH; i++) { g_maze[i] = (int*)calloc(LENGTH, sizeof(int)); } _tprintf(_T("完成!\n")); _tprintf(_T("正在加載迷宮...")); g_lives = 3; for (int j = 0; j < LENGTH; j++) { g_maze[j][0] = ROUTE; g_maze[0][j] = ROUTE; g_maze[j][LENGTH - 1] = ROUTE; g_maze[LENGTH - 1][j] = ROUTE; } _Create(2, 2); g_maze[2][1] = PLAYER; int k; for (k = LENGTH - 3; k >= 0; k--) { if (g_maze[k][LENGTH - 3] == ROUTE) { g_maze[k][LENGTH - 2] = ROUTE; break; } } _tprintf(_T("完成!\n")); _Print(); return k; } void _Print(void) { SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), { 0, 0 }); for (int x = 0; x < LENGTH; x++) { for (int y = 0; y < LENGTH; y++) { switch (g_maze[x][y]) { case ROUTE: _puttch(CH_SPACE); break; case WALL: _puttch(CH_RECT); break; case PLAYER: _puttch(CH_PLAYER); break; default: break; } } _tprintf(_T("\n")); } _putts(_T("上下左右方向鍵用來(lái)移動(dòng),按Esc可退出,按R重新開(kāi)局。")); _tprintf(_T("剩余可撞墻次數(shù):%d"), g_lives); }
這是C++風(fēng)格的代碼。因?yàn)橛玫搅藘?nèi)聯(lián)函數(shù)等C++特性,可能不能直接兼容C語(yǔ)言環(huán)境,但是稍作改動(dòng)即可完美兼容。(ps:至少大家不用為字符集設(shè)置發(fā)愁了 XD)
效果:
其中LENGTH宏規(guī)定了邊長(zhǎng)。這里是30。
想改邊長(zhǎng),直接更改那個(gè)30就可以了。
其中全局變量g_Rank
規(guī)定了難度,數(shù)值越小難度越大,最小值為0。
也是一樣,改難度直接改這個(gè)就OK。
到此這篇關(guān)于C++實(shí)現(xiàn)控制臺(tái)隨機(jī)迷宮的示例代碼的文章就介紹到這了,更多相關(guān)C++ 控制臺(tái)隨機(jī)迷宮內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言 詳細(xì)解析時(shí)間復(fù)雜度與空間復(fù)雜度
算法復(fù)雜度分為時(shí)間復(fù)雜度和空間復(fù)雜度。其作用: 時(shí)間復(fù)雜度是度量算法執(zhí)行的時(shí)間長(zhǎng)短;而空間復(fù)雜度是度量算法所需存儲(chǔ)空間的大小2022-04-04c++雙向鏈表操作示例(創(chuàng)建雙向鏈、雙向鏈表中查找數(shù)據(jù)、插入數(shù)據(jù)等)
這篇文章主要介紹了c++雙向鏈表操作示例,包括創(chuàng)建雙向鏈、刪除雙向鏈表、雙向鏈表中查找數(shù)據(jù)、插入數(shù)據(jù)等,需要的朋友可以參考下2014-05-05C++實(shí)現(xiàn)LeetCode(163.缺失區(qū)間)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(163.缺失區(qū)間),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07Matlab實(shí)現(xiàn)鼠標(biāo)光標(biāo)變成愛(ài)心和瞄準(zhǔn)鏡形狀
這篇文章主要為大家詳細(xì)介紹了如何利用Matlab實(shí)現(xiàn)將鼠標(biāo)光標(biāo)變成愛(ài)心和瞄準(zhǔn)鏡等形狀,文中的示例代碼講解詳細(xì),感興趣的可以了解一下2022-08-08C++基于OpenCV實(shí)現(xiàn)手勢(shì)識(shí)別的源碼
這篇文章主要介紹了C++基于OpenCV手勢(shì)識(shí)別的實(shí)現(xiàn)源碼,這里用到背景減法模型知識(shí),具體實(shí)例代碼跟隨小編一起看看吧2021-09-09