我全程使用TCHAR系列函数,亲测可以不改动代码兼容Unicode/ANSI开发环境,功能正常。大概有100行代码是来自网络的,我也做了改动,侵权请联系删除。本文作者szx0427,只发布于CSDN博客园

这个代码不能算是完美,还是会有轻微的闪屏现象,懒得再加双缓存了,大家可以自行修改。这里用的是SetConsoleCursorPosition函数和cls刷新屏幕。

好了,上代码!VS2015编译通过无警告。其他版本应该也没问题

// 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("你撞墙次数超过限制,本局游戏失败!"));
_putts(_T("如果要再开局,请按下[R]键!否则,按下其他键以退出!"));
ch = _gettch();
if (ch == 'R' || ch == 'r') {
goto start;
}
} else if (bWin) {
_tsystem(_T("cls"));
_putts(_T("恭喜,你赢了!是否要再来一局?"));
_putts(_T("如果要再开局,请按下[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("正在分配内存...")); 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("上下左右方向键用来移动,按Esc可退出,按R重新开局。"));
_tprintf(_T("剩余可撞墙次数:%d"), g_lives);
}

这是C++风格的代码。因为用到了内联函数等C++特性,可能不能直接兼容C语言环境,但是稍作改动即可完美兼容。(ps:至少大家不用为字符集设置发愁了 XD)

效果:

其中LENGTH宏规定了边长。这里是30。



想改边长,直接更改那个30就可以了。

其中全局变量g_Rank规定了难度,数值越小难度越大,最小值为0。

也是一样,改难度直接改这个就OK。


有什么问题和建议,欢迎大家指正!!!!

[开源]C++实现控制台随机迷宫的更多相关文章

  1. 利用纯JS和HTML Canvas生成随机迷宫过程中产生的有趣的事情

    上效果图: #先看生成随机迷宫的代码吧↓ <html> <head> <title>生成随机迷宫v1.0</title> </head> & ...

  2. 斯坦福开源无Bug的随机计算图Certigrad

    斯坦福开源无Bug的随机计算图Certigrad https://news.cnblogs.com/n/573690/ ttps://github.com/dselsam/certigrad

  3. 随机Prim法创建随机迷宫(C#实现)

    因为这两天想参加一个比赛,所以就在上网找素材,刚好看到了迷宫生成,就决定拿这个开刀了. 参考的原文地址为(来源页面) 源地址中是使用AS实现的,没学过AS,所以直接不会运行,于是就自己根据原文的概念进 ...

  4. C++随机迷宫生成[转载]

    原文:http://tieba.baidu.com/p/2596809144 #include<iostream.h> #include"time.h" #includ ...

  5. Unity_Dungeonize 随机生成迷宫

    本文对随机生成迷宫的实现思路进行记录,其作用在于为游戏过程提供随机性以及节省开发周期,下面是Dungeonize的结构 随机迷宫的生成主要包括几个阶段 1.生成房间体结构,为墙体,自定义房间,自定义物 ...

  6. AI-随机迷宫&迷宫求解

    本文记录了,人工智能中简单的搜索策略中的路径搜索策略中的A*算法,来实现迷宫寻路的问题.(这只是一次本人的课外作业) 完整的程序源码已经发送到我的Git.这里只记录了我的思路和感想以及收获. 产生随机 ...

  7. 小项目特供 简易迷宫(基于Java)

    明天返校,于是昨天和今天简单熟系了一下JAVA的GUI,做了一个简易的迷宫小游戏(暂时没有时间实现随机迷宫及多关卡,仅供学习) 源码及运行文件(提供JRE8):链接:简易迷宫 密码:hy8v

  8. Linux下随机生成密码的命令总结

    有时候经常为如何设置一个安全.符合密码复杂度的密码而绞尽脑汁,说实话,这实在是一个体力活而且浪费时间,更重要的是设置密码的时候经常纠结.终于有一天实在忍不住了,于是学习.整理了一下如何使用Linux下 ...

  9. 企业安全建设之搭建开源SIEM平台

    https://www.freebuf.com/special/127172.html https://www.freebuf.com/special/127264.html https://www. ...

随机推荐

  1. LCA总结

    作为一名合格的 OIer ,一定要有自我总结的意识,一定要通过写博客的方式来验证自己的掌握程度 ----沃.茨基硕德 目录 作为一名合格的 OIer ,一定要有自我总结的意识,一定要通过写博客的方式来 ...

  2. Linux + .net core 开发升讯威在线客服系统:同时支持 SQL Server 和 MySQL 的实现方法

    前段时间我发表了一系列文章,开始介绍基于 .net core 的在线客服系统开发过程. 有很多朋友一直提出希望能够支持 MySQL 数据库,考虑到已经有朋友在用 SQL Server,我在升级的过程中 ...

  3. 重新整理 .net core 实践篇————配置中心[四十三]

    前言 简单整理一下配置中心. 正文 什么时候需要配置中心? 多项目组并行协作 运维开发分工职责明确 对风险控制有更高诉求 对线上配置热更新有诉求 其实上面都是套话,如果觉得项目不方便的时候就需要用配置 ...

  4. Spring:Spring-AOP简介

    什么是SpringAOP? 将一些相关的编程方法,独立提取出来,独立实现,然后动态地将代码切入到类的指定方法.指定位置上的编程方式就是AOP(面向切面编程). 讲解一下AOP中的相关概念 Aspect ...

  5. 查看python的安装版本,位数及安装路径

    一.想要查看ubuntu中安装的Python路径 方法一:whereis python (用来快速查找任何文件,是一个文件搜索命令,与locate的功能一样.执行whereis python 会将所有 ...

  6. linux之软连接 硬链接 link ln

    p.p1 { margin: 0; font: 12px "Helvetica Neue"; color: rgba(220, 161, 13, 1) } p.p2 { margi ...

  7. leetcode156场周赛5206

    思路分析: 1.两个数组,一个保存字符,一个保存字符出现次数 2.遍历一遍字符串,出现相同的字符,次数加一,且次数到k的话,那么就剔除,没到k,就次数加一.如果不同,就把它加入到字符的数组里面,对应次 ...

  8. 由ctf来看java报错的危害

    很多java报错在我们渗透的时候经常会被发现,但由于没什么用,危害比较低被忽略,开发也很不愿意修改. 但从纵深防御的角度来说,多个小问题的结合就会产生严重的问题.此次遇到的一个ctf题就是一个例子. ...

  9. Selenium启动Firefox示例(python版)

    目前做selenium自动化使用的主流语言分为java和python,前一篇为java版,本篇介绍python实现selenium启动Firefox. 1 #-*- coding:utf-8 -*- ...

  10. C语言:赋值流程图