游戏中的过程生成——元胞自动机 Celluar Automata 生成洞穴地形
最近在学习过程生成技术,在这里写一些心得。
对于元胞自动机,我们这里只讨论输入是一副二维bool数组的情况,即大多数游戏中的情况。
一个元胞自动机,对于一个输入,给出一个同样格式的输出。输出中的每个点都是按照自动机中的规则从输入中演化而来的。大部分情况下,一个输出上的点,是根据输入中该点周围的点的状态来决定输出中的状态。
我们输入一个二维bool数组,并且规定它的演化规则为,某个点周围的true较多,就演化为true,否则演化为false。这种规则下,我们输入一个随机生成的噪点图,经过一定次数的演化,就可以获得相当自然的“洞穴”地图。
根据噪点图中噪点的数量,最终生成的洞穴被填充的面积也不同。
unity官方给出了一个用元胞自动机随机生成洞穴地形的教程 https://unity3d.com/cn/learn/tutorials/projects/procedural-cave-generation-tutorial
其中的元胞自动机生成的部分代码有一点错误,它并没有把输入的数组进行备份,直接对每个点进行遍历,因此每个点取到的之前一行的点都是新生成的点。
不过负负得正的是因此生成了看起来更像洞穴的结果。事实上如果按照它的规则,写一个周围有4个true,则下一次变为true,否则变为false的自动机的话,最终的图像会出现非常明显的锯齿抖动。如下图。

上图中的锯齿不管进行几次迭代都会一直存在
我做出的修正是当周围的格子为true时刚好有4个时,则自身保持不变。最终获得了看起来像是洞穴的效果。

类似这样的小调整可以做很多,比如边界上的点的处理等。通过不断调整可以获得各种不同效果。
贴一下代码
using UnityEngine;
using System.Collections;
namespace CS.MapGeneration {
public class CelluarAutomata {
private static int[,] offset = {
{ -1, -1 }, { -1, 0 }, { -1, 1 },
{0,-1 }, {0,1 },
{1,-1 }, {1,0 }, {1,1 }
};
private static int GetNeighbourCount(bool[,] map,int x,int y) {
int res = 0;
for (int i = 0; i < offset.GetLength(0); i++) {
int nx = x + offset[i, 0];
int ny = y + offset[i, 1];
if (nx<=0 || ny <=0 || nx>=map.GetLength(0) || ny >= map.GetLength(1) ||map[nx,ny])
res++;
}
return res;
}
public void Iterate(bool[,] map) {
bool[,] copy = map.Clone() as bool[,];
for(int i = 0; i < copy.GetLength(0); i++) {
for(int j = 0; j < copy.GetLength(1); j++) {
map[i, j] = Rule(copy, i, j);
}
}
}
protected static bool Rule(bool[,] map, int x,int y) {
if (!map[x, y] && GetNeighbourCount(map, x, y) > 4)
return true;
else if (map[x, y] && GetNeighbourCount(map, x, y) < 4)
return false;
return map[x, y];
}
}
}
游戏中的过程生成——元胞自动机 Celluar Automata 生成洞穴地形的更多相关文章
- 简单二维元胞自动机 MATLAB实现
20世纪50年代,乌尔姆和冯·诺依曼(对此人真是崇拜的五体投地)为了研究机器人自我复制的可能性,提出了一种叫做元胞自动机(Cellular Automaton,CA)的算法.该算法采用局相互作用规则, ...
- 用C++实现的元胞自动机
我是一个C++初学者,控制台实现了一个元胞自动机. 代码如下: //"生命游戏"V1.0 //李国良于2017年1月1日编写完成 #include <iostream> ...
- 美国康奈尔大学BioNB441元胞自动机MATLAB应用
美国康奈尔大学BioNB441在Matlab中的元胞自动机 介绍 元胞自动机(CA)是用于计算计划利用当地的规则和本地通信.普遍CA定义一个网格,网格上的每个点代表一个有限数量的状态中的细胞.过渡规则 ...
- 基于元胞自动机NaSch模型的多车道手动-自动混合驾驶仿真模型的Matlab实现
模型的建立基本来自于:http://www.doc88.com/p-2078634086043.html 花了一天半的时间用新学会的matlab实现了一下. ───────────────────── ...
- 《 .NET并发编程实战》扩展阅读 - 元胞自动机 - 1 - 为什么要学元胞自动机
先发表生成URL以印在书里面.等书籍正式出版销售后会公开内容.
- matlab学习笔记12单元数组和元胞数组 cell,celldisp,iscell,isa,deal,cellfun,num2cell,size
一起来学matlab-matlab学习笔记12 12_1 单元数组和元胞数组 cell array --cell,celldisp,iscell,isa,deal,cellfun,num2cell,s ...
- MATLAB元胞数组
MATLAB元胞数组 元胞数组: 元胞数组是MATLAB的一种特殊数据类型,可以将元胞数组看做一种无所不包的通用矩阵,或者叫做广义矩阵.组成元胞数组的元素可以是任何一种数据类型的常数或者常量,每一个元 ...
- matlab中元胞数组(cell)转换为矩阵
matlab中元胞数组(cell)转换为矩阵. cell转换为矩阵函数为:cell2mat(c),其中c为待转换的元胞数组: 转化之后的矩阵可能不满足我们对矩阵维数的要求,那么也许还需要下面两个函数: ...
- MATLAB中TXT数据文件读取并写入元胞数组的方法与步骤
一. TXT数据文件读取 Data = load('train.txt'); %简单的文件读取,这时在工作区可以看到导入的大数据变量Data 二.大数据变量Data装入元胞数组中 D = cell ...
随机推荐
- 解析车辆VIN码识别(车架号识别)系统
很多人在购买车辆的时候,只关注性能.外观.内饰等,其实真正的内行是首先看车辆的VIN码,也叫车架号码. VIN码(车架号码)是一辆车的唯一身份证明,一般在车辆的挡风玻璃处,有的在车辆防火墙上,或B柱铭 ...
- 01-Mysql数据库----前戏
MySql的前戏 在学习Mysql之前,我们先来想一下一开始做的登录注册案例,当时我们把用户的信息保存到一个文件中: #用户名 |密码root|123321 alex|123123 上面文件内容的规则 ...
- %matplotlib inline
整理摘自 https://zhidao.baidu.com/question/1387744870700677180.html %matplotlib inline是jupyter notebook里 ...
- 容器基础(八): 使用docker swarm部署程序
环境 基于上一节的env/server:v0.1, env/worker:v0.1镜像, 在基于debian8.2的两台机器上测试部署docker swarm. docker service部署 ➜ ...
- Android java.lang.NoClassDefFoundError的错误
在开发过程中,遇到一个这样的问题:java.lang.NoClassDefFoundError: android.support.v4.util.SparseArrayCompat,这个问题很奇怪,J ...
- Hadoop世界中的HelloWorld之WordCount具体分析
MapReduce 应用举例:单词计数 WorldCount可以说是MapReduce中的helloworld了,下面来看看hadoop中的例子worldcount对其进行的处理过程,也能对mapre ...
- systemtap get var of the tracepoing
kernel.trace("sched_switch") func:func:perf_trace_sched_stat_template get the function in ...
- js & disabled mouse right button menus
js & disabled mouse right button menus 网页可以屏蔽 F12 https://www.cnblogs.com/Marydon20170307/p/9122 ...
- 【bzoj3362/3363/3364/3365】[Usaco2004 Feb]树上问题杂烩 并查集/树的直径/LCA/树的点分治
题目描述 农夫约翰有N(2≤N≤40000)个农场,标号1到N,M(2≤M≤40000)条的不同的垂直或水平的道路连结着农场,道路的长度不超过1000.这些农场的分布就像下面的地图一样, 图中农场用F ...
- [洛谷P3153] [CQOI2009]跳舞
题目大意:有n个女生,n个男生,每次一男一女跳舞.同一队只会跳一次.每个男孩最多只愿意和k个不喜欢的女孩跳舞,女孩同理.问舞会最多能有几首舞曲? 题解:二分跳了多少次舞,每次重建图,建超级原点和汇点, ...