Prim算法生成迷宫
初始化地图
function initMaze(r,c){
let row = new Array(2 * r + 1)
for(let i = 0; i < row.length; i++){
let column = new Array(2 * c + 1)
row[i] = column
for(let j = 0; j < column.length; j++){
row[i][j] = 1
}
}
for(let i = 0; i < r; i++){
for(let j = 0; j < c; j++){
row[2 * i + 1][2 * j + 1] = 0
}
}
console.log(row)
}
initMaze(3,3)
计算二维数组坐标位置
let arr = [
[0,0,0],
[0,0,0],
[0,0,0]
]
for(let i = 0; i < 9; i++){
let row = Math.floor(i / 3)
let column = i % 3
arr[row][column] = false
}
console.log(arr)
偏移量方向预制
let offset = [
{
x:-1,
y:0
},
{
x:1,
y:0
},
{
x:0,
y:-1
},
{
x:0,
y:1
}
]
let x = 0
let y = 0
for(let i = 0; i < offset.length; i++){
x = x + offset[i].x
y = y + offset[i].y
console.log(x,y)
}
随机数公式
1. 0-x之间的随机数:
Math.round(Math.random()*x);
2. x至y之间的随机数
Math.round(Math.random()*(y-x)+x);
3. 1-x之间的随机数:
Math.ceil(Math.random()*x);
Prim算法
const INF = Number.MAX_SAFE_INTEGER
function findMinKey(graph,key,visited){
let min = INF
let minIndex
//找到候选边中成本最小的节点
for(let v = 0; v < graph.length; v++){
if(!visited[v] && key[v] < min){
min = key[v]
minIndex = v
}
}
return minIndex
}
const prim = (graph) => {
const visited = [],
key = [],
parent = [];
let {length} = graph;
for(let v = 0; v < length; v++){
visited[v] = false
key[v] = INF
}
//把到第一个顶点的权值初始化为0
key[0] = 0
parent[0] = -1
//根节点不需要比
for(let i = 0; i < length - 1; i++){
//找到成本最小边的顶点
let u = findMinKey(graph,key,visited)
//标记下该顶点已被访问
visited[u] = true;
//以及该顶点到对应其他顶点是不是成本最小的
//如果是那么就走到该顶点去
for(let v = 0; v < length; v++){
if(graph[u][v] && !visited[v] && graph[u][v] < key[v]) {
parent[v] = u
key[v] = graph[u][v]
}
}
}
return parent
}
const graph = [
[0,2,4,0,0,0],
[2,0,2,4,2,0],
[4,2,0,0,3,0],
[0,4,0,0,3,2],
[0,2,3,3,0,2],
[0,0,0,2,2,0]
]
const parent = prim(graph)
console.log('Edge Weight')
for (let i = 1; i < graph.length; i++) {
console.log(parent[i] + ' - ' + i + ' ' + graph[i][parent[i]]);
}
使用Prim算法生成迷宫
- 生成2 * k + 1的迷宫,1表示墙,0表示路
- 随机选一个顶点,在该顶点上下左右随机抽取一个位置,如果没有访问过而且没有越界就选这个点生成迷宫
- 重复第2步
function roadmap(r,c){
let map = []
let rLen = 2 * r + 1
let cLen = 2 * c + 1
for(let i = 0; i < rLen; i++){
map[i] = []
for(let j = 0; j < cLen; j++){
//生成0101的格式
if((i ^ (i - 1)) === 1 && (j ^ (j - 1)) === 1){
map[i][j] = 0
}else{
map[i][j] = 1
}
}
}
map[1][0] = 0;
map[2 * r - 1][2 * c] = 0;
return map
}
const MathUtil = {
randomInt(a = 0,b){
if(typeof b === 'undefined'){
return Math.floor(Math.random() * a)
} else {
return Math.floor(Math.random() * (b - a) + a)
}
}
}
function maze(map){
let row = map.length >> 1
let col = map[0].length >> 1
let size = row * col
let notAccessed = new Array(size).fill(0)
let accessed = []
let cur = MathUtil.randomInt(0,size)
let offsS = [-col,col,-1,1] //cur在notAccessed要走的偏移量
let offsR = [-1,1,0,0] //cur在map中row要走的偏移量
let offsC = [0,0,-1,1]//cur在map中col要走的偏移量
accessed.push(cur)
notAccessed[cur] = 1
while(accessed.length < size){
let tr = Math.floor(cur / row)
let tc = cur % col
let num = 0;
let pos = -1
//判断四周格子是不是可以走
while(num++ < 4){
let dir = MathUtil.randomInt(0,4)
nr = tr + offsR[dir]
nc = tc + offsC[dir]
if(nr >= 0 && nc >= 0 && nr < row && nc < col && notAccessed[cur + offsS[dir]] === 0){
pos = dir
break
}
}
if(pos < 0){
//堵死的情况
cur = accessed[MathUtil.randomInt(0,accessed.length)]
}else{
//可以走的情况
tr = 2 * tr + 1
tc = 2 * tc + 1
map[tr + offsR[pos]][tc + offsC[pos]] = 0
cur = cur + offsS[pos]
notAccessed[cur] = 1
accessed.push(cur)
}
}
return map
}
function geMaze(r,c){
return maze(roadmap(r,c))
}
Prim算法生成迷宫的更多相关文章
- C++编程练习(10)----“图的最小生成树“(Prim算法、Kruskal算法)
1.Prim 算法 以某顶点为起点,逐步找各顶点上最小权值的边来构建最小生成树. 2.Kruskal 算法 直接寻找最小权值的边来构建最小生成树. 比较: Kruskal 算法主要是针对边来展开,边数 ...
- 图解最小生成树 - 普里姆(Prim)算法
我们在图的定义中说过,带有权值的图就是网结构.一个连通图的生成树是一个极小的连通子图,它含有图中全部的顶点,但只有足以构成一棵树的n-1条边.所谓的最小成本,就是n个顶点,用n-1条边把一个连通图连接 ...
- 图的最小生成树prim算法模板
用prim算法构建最小生成树适合顶点数据较少而边较多的图(稠密图) prim算法生成连通图的最小生成树模板伪代码: G为图,一般为全局变量,数组d为顶点与集合s的最短距离 Prim(G, d[]){ ...
- 最小生成二叉树-prim算法
1.prim算法:一种计算生成最小生成树的方法,它的每一步都会为一棵生长中的树添加一条边. 2.时间复杂度:
- php生成迷宫和迷宫寻址算法实例
较之前的终于有所改善.生成迷宫的算法和寻址算法其实是一样.只是一个用了遍历一个用了递归.参考了网上的Mike Gold的算法. <?php //zairwolf z@cot8.com heade ...
- PHP树生成迷宫及A*自己主动寻路算法
PHP树生成迷宫及A*自己主动寻路算法 迷宫算法是採用树的深度遍历原理.这样生成的迷宫相当的细,并且死胡同数量相对较少! 随意两点之间都存在唯一的一条通路. 至于A*寻路算法是最大众化的一全自己主动寻 ...
- UWP开发:自动生成迷宫&自动寻路算法(3)
+ , + ];//0<=x<=12 0<=y<=24 private static Random Rd = new Random(); 首先声明mazeMap存储数据,声明了 ...
- 生成最小树prim算法
最小生成树prim算法实现 ‘ ’最小生成树,就是权值(两点间直线的值)之和的最小值. 首先,要用二维数组记录点和权值.如上图所示无向图: int G[6][6]; G[1] ...
- 图的生成树(森林)(克鲁斯卡尔Kruskal算法和普里姆Prim算法)、以及并查集的使用
图的连通性问题:无向图的连通分量和生成树,所有顶点均由边连接在一起,但不存在回路的图. 设图 G=(V, E) 是个连通图,当从图任一顶点出发遍历图G 时,将边集 E(G) 分成两个集合 T(G) 和 ...
随机推荐
- B+树索引结构解析
一.二分查找法 二分查找法(binary search)也成为折半查找法.用来查找一组有序的记录组中的某一记录. 基本思想是:将记录按有序化(递增或递减)排列,在查找过程中采用跳跃式方法查找,即先以有 ...
- [ARC083]Collecting Balls
Description 有一个 \(n\times n\) 的矩阵,矩阵内有 \(2n\) 个球.对于 \(i \in [1,n]\) ,\((0,i) (i,0)\) 的位置各有一个启动后往右走/往 ...
- SpringBoot项目FatJar同War之间的相互转换
目录 1.原有的项目结构 修改后的项目结构 常见的dependency的scope 1.原有的项目结构 启动类 @SpringBootApplication(exclude = {DataSource ...
- 最短路径(floyd和Dijkstra)
最短路 Time Limit: 5000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...
- mysql双主+keepalived架构
架构展示 操作系统 centos6.5 数据库 mysql5.7 master1 10.0.254.148 master2 10.0.254.147 VIP 10.0.254.88 (keepaliv ...
- PHP实现上传文件到服务器
<?php /**************************** *** 功能:上传文件到服务器 ****************************/ session_start() ...
- k3 cloud库存管理中的直接调拨单权限分配出现问题
k3 cloud中给直接调拨单分配了对应的权限,但是客户端无法查看到对应的单据 解决办法: 是应为没有发布到对应的客户端和浏览器端,打开bos,找到对应的单据并点击发布,找到对应的目录,如图所示: 把 ...
- JavaScript设计模式-组合模式(表单应用实现)
书读百遍其义自见 <JavaScript设计模式>一书组合模式在表单中应用,我问你答答案. 注:小编自己根据书中的栗子码的答案,如有错误,请留言斧正. 另:如有转载请注明出处,谢谢啦 &l ...
- [JavaScript深入系列]JavaScript深入之执行上下文栈(转载)
顺序执行? 如果要问到 JavaScript 代码执行顺序的话,想必写过 JavaScript 的开发者都会有个直观的印象,那就是顺序执行,毕竟: var foo = function () { co ...
- 2019-9-2-win10-UWP-MvvmLight入门
title author date CreateTime categories win10 UWP MvvmLight入门 lindexi 2019-09-02 12:57:38 +0800 2018 ...