最近在学习过程生成技术,在这里写一些心得。

对于元胞自动机,我们这里只讨论输入是一副二维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 生成洞穴地形的更多相关文章

  1. 简单二维元胞自动机 MATLAB实现

    20世纪50年代,乌尔姆和冯·诺依曼(对此人真是崇拜的五体投地)为了研究机器人自我复制的可能性,提出了一种叫做元胞自动机(Cellular Automaton,CA)的算法.该算法采用局相互作用规则, ...

  2. 用C++实现的元胞自动机

    我是一个C++初学者,控制台实现了一个元胞自动机. 代码如下: //"生命游戏"V1.0 //李国良于2017年1月1日编写完成 #include <iostream> ...

  3. 美国康奈尔大学BioNB441元胞自动机MATLAB应用

    美国康奈尔大学BioNB441在Matlab中的元胞自动机 介绍 元胞自动机(CA)是用于计算计划利用当地的规则和本地通信.普遍CA定义一个网格,网格上的每个点代表一个有限数量的状态中的细胞.过渡规则 ...

  4. 基于元胞自动机NaSch模型的多车道手动-自动混合驾驶仿真模型的Matlab实现

    模型的建立基本来自于:http://www.doc88.com/p-2078634086043.html 花了一天半的时间用新学会的matlab实现了一下. ───────────────────── ...

  5. 《 .NET并发编程实战》扩展阅读 - 元胞自动机 - 1 - 为什么要学元胞自动机

    先发表生成URL以印在书里面.等书籍正式出版销售后会公开内容.

  6. 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 ...

  7. MATLAB元胞数组

    MATLAB元胞数组 元胞数组: 元胞数组是MATLAB的一种特殊数据类型,可以将元胞数组看做一种无所不包的通用矩阵,或者叫做广义矩阵.组成元胞数组的元素可以是任何一种数据类型的常数或者常量,每一个元 ...

  8. matlab中元胞数组(cell)转换为矩阵

    matlab中元胞数组(cell)转换为矩阵. cell转换为矩阵函数为:cell2mat(c),其中c为待转换的元胞数组: 转化之后的矩阵可能不满足我们对矩阵维数的要求,那么也许还需要下面两个函数: ...

  9. MATLAB中TXT数据文件读取并写入元胞数组的方法与步骤

    一. TXT数据文件读取 Data = load('train.txt');   %简单的文件读取,这时在工作区可以看到导入的大数据变量Data 二.大数据变量Data装入元胞数组中 D = cell ...

随机推荐

  1. Get Error when restoring database in Sql Server 2008 R2

         When I restored a database I got an error: "The backup set holds a backup of a database ot ...

  2. C++学习011-常用内存分配及释放函数

    C++用有多种方法来分配及释放内存,下面是一些经常使用的内存分配及释放函数 现在我还是一个技术小白,一般用到也指示 new+delete 和 malloc和free 其他的也是在学习中看到,下面的文字 ...

  3. Bugku 速度要快

    import requests import base64 url="http://123.206.87.240:8002/web6/" res=requests.get(url) ...

  4. tomcat8编码设置和gc异常解决

    用startup.bat启动 编码解决: 用编辑器打开catalina.bat文件找到set "JAVA_OPTS=%JAVA_OPTS% %JSSE_OPTS% " 更改为 se ...

  5. HDFS分布式集群

    一.HDFS伪分布式环境搭建 Hadoop分布式文件系统(HDFS)被设计成适合运行在通用硬件(commodity hardware)上的分布式文件系统.它和现有的分布式文件系统有很多共同点.但同时, ...

  6. P3950部落冲突

    题面 \(Solution:\) 法一:LCT裸题 又好想又好码,只不过常数太大. 法二:树链剖分 每次断边将该边权的值++,连边--,然后边权化点权(给儿子),询问就查询从x到y的路径上的边权和,树 ...

  7. kaldi解码及特征提取详解

    目录 1. 注意事项 2. 流程图: 3. 具体流程指令: 1. 注意事项 首先要训练好模型,用到3个文件,分别是: final.mdl(训练模型得到的模型文件) final.mat(用来特征转换) ...

  8. NHibernate3.3.3 学习笔记1

    前言 昨天在园友的介绍下,我找了一本学习NHibernate的书:<NHibernate 3 Beginner’s Guide>. 第一章我直接跳过了,因为是英文版的看起来很吃力,且第一章 ...

  9. PAT 1040 有几个PAT

    https://pintia.cn/problem-sets/994805260223102976/problems/994805282389999616 字符串 APPAPT 中包含了两个单词 PA ...

  10. To Chromium之版本管理

    Git. 1.由于想直接submit到Chromium的官方Branch需要申请权限,目前拿不到,所以打算snapshot一个chromium版本. 本地搭建一个git的server/client,方 ...