c#元胞自动机_游戏中的过程生成——元胞自动机 Celluar Automata 生成洞穴地形

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

对于元胞自动机,我们这里只讨论输入是一副二维bool数组的情况,即大多数游戏中的情况。

一个元胞自动机,对于一个输入,给出一个同样格式的输出。输出中的每个点都是按照自动机中的规则从输入中演化而来的。大部分情况下,一个输出上的点,是根据输入中该点周围的点的状态来决定输出中的状态。

我们输入一个二维bool数组,并且规定它的演化规则为,某个点周围的true较多,就演化为true,否则演化为false。这种规则下,我们输入一个随机生成的噪点图,经过一定次数的演化,就可以获得相当自然的“洞穴”地图。

根据噪点图中噪点的数量,最终生成的洞穴被填充的面积也不同。

其中的元胞自动机生成的部分代码有一点错误,它并没有把输入的数组进行备份,直接对每个点进行遍历,因此每个点取到的之前一行的点都是新生成的点。

不过负负得正的是因此生成了看起来更像洞穴的结果。事实上如果按照它的规则,写一个周围有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];

}

}

}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值