O(n)线性空间的迷宫生成算法
之前所有的迷宫生成算法,空间都是O(mn),时间同样是O(mn),时间上已经不可能更优化,
于是,我就从空间优化上着手,研究一个仅用O(n)空间的生成算法。
我初步的想法是,每次生成一行,生成后立即输出,而其连通性的信息用并查集保存。
然而这时却遇到阻力:不可能简单保存连通性信息就能完成。
因为通过上一行的信息去生成下一行的时候,仅靠连通性信息你无法知道你能不能结束当前路径上的连结,
也就是说你不能判断你能不能生成一个死胡同。
后来想了一下,改进了一下并查集,让它的父结点多保存一个信息:当前行与本结点连通的格子数
看这个例子:
━┳━━┳━━┳━━━━┳━━┳━━┳┓
┣┓┗━┃┃━━┛━━━┓┃┃┃┗┓┃┃┃
┃┗┳━┫┗━━┓┏━━┫┏┛┣━┃┃┃┃
1 1 1 1 2 2 2 2 3 6 6 6 3 3 3 3 3 4 4 5
相同数字的表示它们连通(连向同一个父结点),比如3那个,当前行与本结点连通的格子数=6
也就是3出现的次数。
在生成下一行的时候,每生成一个格子,就得同时更新连通格子数的值,以保证不会生成环路和死路
参考源代码:
#include <iostream>
#include <ctime>
using namespace std;
#define next(n,s) ((n)=(n)%((s)-1)+1)
int fset_size;
struct data
{
int parent;
int level;
int sum;
}*fset;
int fset_pt = 1, fdeep;
int try_alloc(int level)
{
if (fset[fset_pt].level < level)
{
return fset_pt;
}
int lpt = fset_pt;
for (next(fset_pt, fset_size); fset_pt!=lpt; next(fset_pt, fset_size))
{
if (fset[fset_pt].level < level)
{
return fset_pt;
}
}
return 0;
}
int reg_fset(int level)
{
int lpt = fset_pt;
fset[fset_pt].level = level;
fset[fset_pt].parent = 0;
fset[fset_pt].sum = 1;
next(fset_pt, fset_size);
return lpt;
}
int get_parent(int id)
{
int d = id;
for (fdeep=0; fset[id].parent>0; ++fdeep)
id = fset[id].parent;
if (d != id) fset[d].parent = id;
return id;
}
void insert(int a, int b)
{
int pa = get_parent(a), da = fdeep;
int pb = get_parent(b), db = fdeep;
if (pa==pb)
{
}
else if (da<=db)
{
fset[pa].parent = pb;
fset[pb].level = max(fset[pa].level, fset[pb].level);
fset[pb].sum += fset[pa].sum-1;
}
else
{
fset[pb].parent = pa;
fset[pa].level = max(fset[pa].level, fset[pb].level);
fset[pa].sum += fset[pb].sum-1;
}
}
int is_connect(int a, int b)
{
if (a==b || get_parent(a)==get_parent(b))
return 1;
return 0;
}
struct tile
{
int wall;
int id;
};
char cw[][4]={" ","━","┃","┛","━","━","┗","┻","┃","┓","┃","┫","┏","┳","┣","╋"};
int Gen(int w, int h)
{
int x,y,lx,ly,p;
tile* maze = (tile*)malloc(sizeof(tile)*(w+2));
fset = (data*)malloc(sizeof(data)*(w*2));
fset_size = w;
memset(fset, 0, sizeof(data)*(w*2));
for (x=0; x<=w+1; ++x) maze[x].wall = 7, maze[x].id=0;
maze[0].wall = 15; maze[w+1].wall = 15;
for (y=1; y<=h+1; ++y,maze[0].wall = 11,maze[w+1].wall = 14)
{
lx = 0; ly = 1; x = 0;
fset[0].sum = 0;
p = 15;
if (lx) p &= ~8;
if (ly) p &= ~1;
if (maze[x+1].wall&8) p &= ~4;
if (is_connect(maze[x].id, maze[x+1].id)) p &= ~2;
if (y>h) p &= ~8;
printf(cw[p]);
p &= 4;
printf(cw[p]);
for (x=1; x<=w; ++x)
{
int r, _r = 4;
ly = (maze[x].wall&8);
int id, dsum = 0;
if (lx && ly)
{
insert(maze[x-1].id, maze[x].id);
}
else if (lx==0 && ly==0)
{
maze[x].id = try_alloc(y);
if (maze[x].id==0)
{
fset_size += w;
maze[x].id = try_alloc(y);
fset_size -= w;
}
reg_fset(y+1);
}
else if (lx)
{
maze[x].id = maze[x-1].id;
}
id = get_parent(maze[x].id);
if ((maze[x+1].wall&8) && is_connect(maze[x].id, maze[x+1].id)) _r = 2;
if (y==h && x==w)
{
r = 0;
}
else do
{
r = rand() % _r;
if ((r&2)==0 && try_alloc(y)==0) r |= 2;
if (y>h)
{
r = 2;
}
else if (x==w) r &= ~2;
if (y==h) r &= ~1;
dsum = 0;
if ((r & 1) == 0 ) dsum -= 1;
if (r & 2) dsum += 1;
}
while (y<=h && ((r==0 && (lx==0 && ly==0) || fset[id].sum+dsum<=0 )) );
maze[x].wall = 0;
if (ly) maze[x].wall |= 2;
if (lx)
{
maze[x].wall |= 1;
}
lx = (r&2);
if (lx) maze[x].wall |= 4;
if (r&1) maze[x].wall |= 8;
p = 15;
if (maze[x].wall&4) fset[id].sum += 1;
if ((maze[x].wall&8)==0) fset[id].sum -= 1, fset[0].sum += 1;
fset[id].level = y+1;
if (maze[x].wall&4) p &= ~8;
if (maze[x].wall&2) p &= ~1;
if (maze[x+1].wall&8) p &= ~4;
if (maze[x+1].wall&1) p &= ~2;
printf(cw[p]);
p &= 4;
printf(cw[p]);
}
puts("");
if (y<h)
{
for (x=0; x<=w; ++x)
{
if (maze[x].wall&4) printf(cw[0]); else printf(cw[10]);
int id = get_parent(maze[x].id);
maze[x].id = id;
printf(cw[0]);
}
puts("");
}
else if (y==h)
{
for (x=0; x<=w; ++x)
{
if ((maze[x].wall&4) || x==w) printf(cw[0]); else printf(cw[10]);
int id = get_parent(maze[x].id);
maze[x].id = id;
printf(cw[0]);
}
puts("");
}
}
free(maze);
free(fset);
return 0;
}
int main()
{
srand((unsigned)time(NULL));
Gen(15,10);
return 0;
}
O(n)线性空间的迷宫生成算法的更多相关文章
- [迷宫中的算法实践]迷宫生成算法——递归分割算法
Recursive division method Mazes can be created with recursive division, an algorithm which wo ...
- Clojure——学习迷宫生成
背景 初学clojure,想着看一些算法来熟悉clojure语法及相关算法实现. 找到一个各种语言生成迷宫的网站:http://rosettacode.org/wiki/Maze_generation ...
- php生成迷宫和迷宫寻址算法实例
较之前的终于有所改善.生成迷宫的算法和寻址算法其实是一样.只是一个用了遍历一个用了递归.参考了网上的Mike Gold的算法. <?php //zairwolf z@cot8.com heade ...
- 《C++游戏开发》十六 游戏中的寻路算法(二):迷宫&A*算法基础
本系列文章由七十一雾央编写,转载请注明出处. http://blog.csdn.net/u011371356/article/details/10289253 作者:七十一雾央 新浪微博:http: ...
- roguelike地牢生成算法
文章原地址 上一个地图生成算法,这一次是一个地牢的生成算法,是一个国外的人写的算法,用dart语言写,我把它改成了unity-c#. 原作者博客地址:Rooms and Mazes: A Proced ...
- 一个UUID生成算法的C语言实现 --- WIN32版本 .
一个UUID生成算法的C语言实现——WIN32版本 cheungmine 2007-9-16 根据定义,UUID(Universally Unique IDentifier,也称GUID)在时 ...
- 分布式全局不重复ID生成算法
分布式全局不重复ID生成算法 算法全局id唯一id 在分布式系统中经常会使用到生成全局唯一不重复ID的情况.本篇博客介绍生成的一些方法. 常见的一些方式: 1.通过DB做全局自增操作 优点:简单.高 ...
- Java与算法之(12) - 老鼠再闯迷宫(广度优先算法)
贪吃的小老鼠又回来了,这次有什么新的办法吃到奶酪呢? 规则不变,只能上下左右在格子内移动. 因为上次的深度优先算法让老鼠走了不少冤枉路,这次老鼠带来了帮手探路鼠.探路鼠的使用规则如下: 小老鼠按右.下 ...
- C++ 基于凸包的Delaunay三角网生成算法
Delaunay三角网,写了用半天,调试BUG用了2天……醉了. 基本思路比较简单,但效率并不是很快. 1. 先生成一个凸包: 2. 只考虑凸包上的点,将凸包环切,生成一个三角网,暂时不考虑Delau ...
随机推荐
- Centos7 docker 常用指令
Docker 运行在 CentOS 7 上,要求系统为64位.系统内核版本为 3.10 以上 一.docker的安装及卸载 1.查看当前系统内核版本: [root@docker ~]# uname - ...
- Note: further occurrences of HTTP header parsing errors will be logged at DEBUG level
2018-03-23 18:32:21,690 [INFO] [http-nio-11007-exec-2] org.apache.coyote.http11.Http11Processor [Dir ...
- RabbitMQ脑裂问题解决方案调查
现象: RabbitMQ GUI上显示 Network partition detectedMnesia reports that this RabbitMQ cluster has experien ...
- MySQL查询性能优化---高性能(二)
转载地址:https://segmentfault.com/a/1190000011330649 避免向数据库请求不需要的数据 在访问数据库时,应该只请求需要的行和列.请求多余的行和列会消耗MySql ...
- jsp jstl标签库核心标签
JSTL标签库介绍 JSTL标签库的使用时为了弥补html标签的不足,规范自定义标签的使用而诞生的.使用标签的目的就是不希望在jsp页面中出现java逻辑代码 全称:JSTL标签库分类 核心标签库使用 ...
- [转]VirtualBox中的网络连接方式详解
如果出现主机无法ping通虚拟机的情况,请首先确认虚拟机防火墙已关闭. 一.NAT模式 特点: 1.如果主机可以上网,虚拟机可以上网 2.虚拟机之间不能ping通 3.虚拟机可以ping通主机(此时p ...
- SpingBoot二——引入MySql数据库
◆版权声明:本文出自胖喵~的博客,转载必须注明出处. 转载请注明出处:https://www.cnblogs.com/by-dream/p/10486117.html 搭起一个简单的服务后,接下来我们 ...
- ORACLE telnet 1521 不通及ORA-12514: TNS: 监听程序当前无法识别连接描述符中请求的服务的解决
服务器上安装了oracle11g , 防火墙上已经增加1521 入站规则.但是内网客户端配置好了TNS无法连接.telnet 1521 不通. 需要在服务器上\product\10.2.0\db_1\ ...
- 福大软工1816 - 第八次作业(课堂实战)- 项目UML设计
团队 学号 姓名 本次作业博客链接 031602428 苏路明(组长) https://www.cnblogs.com/Sulumer/p/9822854.html 031602401 陈瀚霖 htt ...
- 2018-2019-2 网络对抗技术 20165202 Exp5 MSF基础应用
博客目录 一.实践目标 二.实践内容 一个主动攻击实践,ms08_067(成功).ms03_026(成功且唯一); 一个针对浏览器的攻击,如ms11_050(成功)ms11_03(失败.唯一)ms10 ...