在JS中预生成伪随机地形

Pre-generating pseudo-random terrain in JS

本文关键字:随机 JS      更新时间:2023-09-26

我正在开发一款简单的JS/画布游戏。在这个游戏中,我希望用户能够导航一个由客户端每次根据相同种子生成的世界。因此,尽管世界是随机的,但每个用户都会得到相同的结果。

因此,我正在寻找一种方法来做到以下几点:

var some_seed = "abcdefg" // For instance
function get_world_rect(ab, cd) { ... }
get_world_rect([0,0], [9,9])
// Yields the following: 
[[0, 0, 0, 0, 0, 0, "some_feature", 0, 0, 0], 
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, "rock", 0, 0, "bush", 0, 0, 0, 0, 0], 
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0], 
 [0, 0, 0, "bush", 0, 0, 0, 0, 0, 0], 
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0], 
 [0, 0, 0, 0, 0, 0, 0, "bush", 0, 0], 
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0], 
 [0, 0, "rare_flower", 0, 0, 0, 0, 0, 0, 0], 
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]]

这里的两个重要部分是,我希望能够在每个客户端上每次生成相同的"地图",并控制某些功能的"稀有性"。所以我可以说,这张地图有X次产卵"bush"的机会,Y次产卵"rare_flower"的机会。

我使用了一个自定义的随机数生成器。然后我给它一个种子加上一些基于瓦片的x和y的计算。

不确定这是否是我能做的最好的,但它有效。

下面显示一个片段。

generate_map = function(chunk_x, chunk_y) {
    // The MersenneTwister object is from code this link: https://gist.github.com/banksean/300494
    // this.base seed is some int.
    // a "map chunk" is a square of tile containing this.map_chunk_size tiles.
    // chunk_x and chunk_y are that chunk's X and Y.
    g = new MersenneTwister(this.base_seed + chunk_x*1000000 + chunk_y)
    tiles = []
    for(var x = 0; x < this.map_chunk_size; x++) {
        for(var y = 0; y < this.map_chunk_size; y++) {
            var k = g.random()
            /* 
            Objects:
            0 - empty
            1 - something 1
            2 - something 2
            */
            var obj = 0
            if (k < 0.001) {
                // Something rare
                obj = 2
            } else if (k < 0.1) {
                obj = 1
            }
            tiles.push(obj)
        }
    }
    return tiles
}

这样,它在浏览器和操作系统之间是可预测的(据我所知),而且我不必在服务器上生成并保存大量数据。

据我所知,对性能的影响微乎其微。