Perlin Noise не набирает глубину с большим количеством октав

Я плохо знаком с шумом Перлина, и я столкнулся с препятствием. Кажется, что функция шума perlin, которую я перевел с C++, работает правильно для одной октавы, однако я обнаружил, что более низкие октавы шума не добавляются к исходному шуму Perlin. Вот мой код:

public class Perlin {
    float[][] generateWhiteNoise(int width, int height) {
        Random random = new Random(0);
        float[][] noise = new float[width][height];

        for (int i = 0; i < noise.length; i++) {
            for (int j = 0; j < noise[i].length; j++){
                noise[i][j] = (float)random.nextDouble();
            }
        }

        return noise;
    }

    float[][] generateSmoothNoise(float[][] baseNoise, int octave){
        int width = baseNoise.length;
        int height = baseNoise[0].length;

        float[][] smoothNoise = baseNoise;

        int samplePeriod = (int) Math.pow(2,octave); // calculates 2 ^ k
        float sampleFrequency = 1.0f / samplePeriod;

        for (int i = 0; i < width; i++) {
          //calculate the horizontal sampling indices
          int sample_i0 = (i / samplePeriod) * samplePeriod;
          int sample_i1 = (sample_i0 + samplePeriod) % width; //wrap around
          float horizontal_blend = (i - sample_i0) * sampleFrequency;
          for (int j = 0; j < height; j++){
             //calculate the vertical sampling indices
             int sample_j0 = (j / samplePeriod) * samplePeriod;
             int sample_j1 = (sample_j0 + samplePeriod) % height; //wrap around
             float vertical_blend = (j - sample_j0) * sampleFrequency;
             //blend the top two corners
             float top = interpolate(baseNoise[sample_i0][sample_j0],
                baseNoise[sample_i1][sample_j0], horizontal_blend);

             //blend the bottom two corners
             float bottom = interpolate(baseNoise[sample_i0][sample_j1],
                baseNoise[sample_i1][sample_j1], horizontal_blend);

             //final blend
             smoothNoise[i][j] = interpolate(top, bottom, vertical_blend);
          }
       }

       return smoothNoise;
    }

    float interpolate(float x0, float x1, float alpha){
        return (float) ((float)(x0) * (float)(1 - alpha) + (float)(alpha * x1));
    }

    float[][] generatePerlinNoise(float[][] baseNoise, int octaveCount) {
       int width = baseNoise.length;
       int height = baseNoise[0].length;

       float[][][] smoothNoise = new float[octaveCount][][]; //an array of 2D arrays containing

       float persistance = .5f;

       //generate smooth noise
       for (int i = 0; i<octaveCount; i++) {
           System.out.println("Generating Smooth Noise: " + i);
           smoothNoise[i] = generateSmoothNoise(baseNoise, i);
       }

        float[][] perlinNoise = new float[width][height];
        float amplitude = 1.0f;
        float totalAmplitude = 0.0f;

        //blend noise together
        for (int octave = octaveCount - 1; octave >= 0; octave--) {
           amplitude *= persistance;
           totalAmplitude += amplitude;
           System.out.println("Adding smooth noise for octave: " + octave + " at amplitude: " + amplitude);
           for (int i = 0; i < width; i++) {
              for (int j = 0; j < height; j++) {
                 perlinNoise[i][j] += smoothNoise[octave][i][j] * amplitude;
              }
           }
        }

       //normalization
       for (int i = 0; i < width; i++) {
          for (int j = 0; j < height; j++) {
             perlinNoise[i][j] /= totalAmplitude;
          }
       }
       return perlinNoise;
    }

    public float[][] printVals(float[][] baseNoise){
        baseNoise = generatePerlinNoise(generateWhiteNoise(800,800),6);
        for(int i = 0; i<baseNoise.length; i++){
            String row = "";
            for(int j = 0; j<baseNoise[i].length;j++){
                row+= (int)(baseNoise[i][j]*255) + " ";
            }
            System.out.println(row);
        }
        return baseNoise;
    }
}

Вот код, который я использую для получения значений:

baseNoise = generatePerlinNoise(generateWhiteNoise(800,800),6);

Вот выходы на октавы 1,2,3 и 4

Частота 1Частота 2Частота 3Частота 4

Любая помощь будет оценена!

Изменить: методом проб и ошибок, я обнаружил, что наиболее вероятная область для проблем в generatePerlinNoise() функция. Если я изменяю октавы, я получаю желаемый уровень шума, который я и хочу. Это также означает, что generateWhiteNoise() а также generateSmoothNoise() работает. Итак, где-то внутри generatePerlinNoise() смешивание есть проблема, но кажется, что это должно работать.

1 ответ

Я нашел ответ. Я думал, что generateSmoothNoise() Команда собиралась дать мне многооктавный перлин-шум. Тем не менее, я понял, что мне нужно создать другую команду, похожую на то, что происходит с наложением плавного шума, чтобы смешать мои разные октавы перлин-шума.

public float[][] generateMultiOctavePerlinNoise(int octaves, double persistence, double dropoff, int width, int height){
        float[][][]noise = new float[octaves][width][height];
        for(int i = octaves - 1; i > 0;i--){
            noise[i] = generatePerlinNoise(generateWhiteNoise(width,height),octaves - i);
        }

        float[][] multiOctave = new float[width][height];

        for(int a= 0; a<noise.length; a++){
            persistence*= dropoff;
            for(int i = 0; i<multiOctave.length; i++){
                for(int j = 0; j<multiOctave[i].length; j++){
                    multiOctave[i][j] += noise[a][i][j]*persistence;
                }
            }
        }
        return multiOctave;

    }
Другие вопросы по тегам