- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我最近一直在尝试在 C 中实现柏林噪声生成器(基于 Ken Perlin's website,使用 SDL 库作为屏幕输出),但输出显示插值 block 之间的边缘不连续或不平滑 -插值 block 确实表现为 block 。
我尝试了四种插值,所有“平滑”的看起来都差不多;相比之下,只有余弦看起来(非常)稍微好一些,直线看起来很糟糕。 (下面是余弦和线性)
具有讽刺意味的是,如果对噪声进行分形求和(我这样做的最终目的),则线性消除了“ block 状”方面的平滑插值,实际上看起来几乎没问题。
我很确定我的代码中缺少某些东西或做错了,但我似乎找不到它。
作为引用,我当前的代码如下:
#include<stdio.h>
#include<math.h>
#include<SDL/SDL.h>
void normalize3(float *vec3){
float distX=0,distY=0,distZ=0;
distX=vec3[0];
distX*=distX;
distY=vec3[1];
distY*=distY;
distZ=vec3[2];
distZ*=distZ;
float dist=sqrtf(distX+distY+distZ);
vec3[0]/=dist;
vec3[1]/=dist;
vec3[2]/=dist;
}
float sinterpolate(float scale){
//return scale*scale*(3.0-2*scale); //Classic 3*t^2-2*t^3
/*float t=scale*scale;
float u=t*t;
return (6.0*u*scale-15.0*u+10.0*t*scale);*/ //Improved 6*t^5-15*t^4+10*t^3
return (0.5-cosf(scale*M_PI)/2.0); //Straight cosine interpolation
}
float linterpolate(float a,float b,float scale){
return a+scale*(b-a);
}
float noise3(float *vec3,float *grads,Uint8 *perms){
vec3[0]=fmodf(vec3[0],256.0);
vec3[1]=fmodf(vec3[1],256.0);
vec3[2]=fmodf(vec3[2],256.0);
Uint8 ivec3[3];
float relPos[3],temp;
float cube[2][2][2];
Uint8 index;
//One loop for each dimension of noise.
for(int x=0;x<2;x++){
ivec3[0]=vec3[0];
ivec3[0]+=x;
relPos[0]=vec3[0]-ivec3[0];
for(int y=0;y<2;y++){
ivec3[1]=vec3[1];
ivec3[1]+=y;
relPos[1]=vec3[1]-ivec3[1];
for(int z=0;z<2;z++){
ivec3[2]=vec3[2];
ivec3[2]+=z;
relPos[2]=vec3[2]-ivec3[2];
index=ivec3[0]+perms[ivec3[1]+perms[ivec3[2]]];
temp=relPos[0]*grads[3*index];
temp+=relPos[1]*grads[3*index+1];
temp+=relPos[2]*grads[3*index+2]; //The gradient's dot product
//with respect to the point
//being analyzed
cube[x][y][z]=temp;
}
}
}
ivec3[0]--;
ivec3[1]--;
ivec3[2]--;
relPos[0]=vec3[0]-ivec3[0];
relPos[1]=vec3[1]-ivec3[1];
relPos[2]=vec3[2]-ivec3[2];
relPos[0]=sinterpolate(relPos[0]); //Comment these
relPos[1]=sinterpolate(relPos[1]); //if you want
relPos[2]=sinterpolate(relPos[2]); //Linear Interpolation.
return linterpolate(linterpolate(linterpolate(cube[0][0][0],cube[0][0][1],relPos[2]),linterpolate(cube[0][8][0], cube[0][9][1],relPos[2]),relPos[1]),linterpolate(linterpolate(cube[1][0][0],cube[1][0][1],relPos[2]),linterpolate(cube[1][10][0], cube[1][11][1],relPos[2]),relPos[1]),relPos[0]);
}
int main(int argc,char **args){
SDL_Init(SDL_INIT_VIDEO);
SDL_Surface *screen=SDL_SetVideoMode(512,512,32,SDL_SWSURFACE);
srandom(SDL_GetTicks()); //If not on OSX/BSD, use srand()
Uint32 *pixels;
Uint32 grays[256];
for(int x=0;x<256;x++){
grays[x]=SDL_MapRGB(screen->format,x,x,x);
}
float grads[768];
Uint8 perms[256];
//First, generate the gradients and populate the permutation indexes.
for(int x=0;x<256;x++){
grads[3*x]=random(); //If not on OSX/BSD, use rand()
grads[3*x+1]=random();
grads[3*x+2]=random();
normalize3(grads+3*x);
perms[x]=x;
}
//Let's scramble those indexes!
for(int x=0;x<256;x++){
Uint8 temp=perms[x];
Uint8 index=random();
perms[x]=perms[index];
perms[index]=temp;
}
printf("Permutation Indexes: ");
for(int x=0;x<256;x++){
printf("%hhu, ",perms[x]);
}
putchar('\n');
Uint32 timer=SDL_GetTicks(),frameDelta;
SDL_Event eventos;
float zoom=-5.0;
eventos.type=SDL_NOEVENT;
while(eventos.type!=SDL_QUIT){
SDL_PollEvent(&eventos);
if(SDL_GetKeyState(NULL)[SDLK_UP]){
zoom-=0.001*frameDelta;
}
else if(SDL_GetKeyState(NULL)[SDLK_DOWN]){
zoom+=0.001*frameDelta;
}
float scale=expf(zoom);
pixels=screen->pixels;
float pos[3];
pos[2]=SDL_GetTicks()/3000.0;
for(int y=0;y<512;y++){
pos[1]=y*scale;
for(int x=0;x<512;x++){
pos[0]=x*scale;
float fracPos[3];
fracPos[0]=pos[0];
fracPos[1]=pos[1];
fracPos[2]=pos[2];
float color=noise3(fracPos,grads,perms);
//Fractal sums of noise, if desired
/*fracPos[0]*=2.0;
fracPos[1]*=2.0;
fracPos[2]*=2.0;
color+=noise3(fracPos,grads,perms)/2.0;
fracPos[0]*=2.0;
fracPos[1]*=2.0;
fracPos[2]*=2.0;
color+=noise3(fracPos,grads,perms)/4.0;
fracPos[0]*=2.0;
fracPos[1]*=2.0;
fracPos[2]*=2.0;
color+=noise3(fracPos,grads,perms)/8.0;
fracPos[0]*=2.0;
fracPos[1]*=2.0;
fracPos[2]*=2.0;
color+=noise3(fracPos,grads,perms)/16.0;
*/
*pixels++=grays[127+(Sint8)(256.0*color)];
}
}
SDL_Flip(screen);
frameDelta=SDL_GetTicks()-timer;
printf("Running @ %.3f FPS!\n",1000.0/frameDelta);
if(frameDelta<16){
SDL_Delay(16-frameDelta);
}
timer=SDL_GetTicks();
}
return 0;
}
用法:在运行时,按住向上或向下可放大或缩小噪声网格。
最佳答案
我终于找到了问题所在:梯度生成器。
我假设 random() 函数会将其二进制值传递给 grads[] 数组,从而覆盖整个 float 范围。不幸的是,情况并非如此:它的返回值首先被转换为 float ,然后存储在数组中。我最大的问题是所有生成的 vector 都具有正成员值。
这证明了方 block 工件的合理性:有许多彼此相邻的“山丘”(高值)生成,但没有“山谷”(低值),并且两个相邻的山丘最终会发生冲突并沿着整数值。
意识到这一点后,我尝试做一些指针杂耍并直接以 Uint32 形式存储值,但是渐变中的值变得古怪(infs、NaNs、1.0s 和 0.0s),所以我回来了到原始路线并否定代码本身中的数字。
这个 7-liner 解决了整个问题:
int y=random()&7;
if(y&1)
grads[3*x]*=-1.0f;
if(y&2)
grads[3*x+1]*=-1.0f;
if(y&4)
grads[3*x+2]*=-1.0f;
只需将其放在归一化函数之前或之后即可完成。
现在它看起来像 Perlin Noise:
分形和看起来也好一点:
@DiJuMx:我以前看过“改善噪声”论文,但没有意识到梯度对噪声外观的影响有多大。此外,通过尝试将坐标空间从 0~256 更改为 0~1 导致分形和不再起作用,并且生成的图像具有相同的 block 伪影。
关于c - "Blocky"柏林噪声,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14206861/
我正在研究我的论文,以构建乐谱(乐谱)识别系统。这是我的输入文件: 这是一些预处理和去除五线谱后的结果: 我在这里遇到的问题是,在五线谱删除步骤之后出现了一些超小的、不需要的“点”。这些点/噪声与实际
这可能是一个愚蠢的问题(当然),但对于我已阅读/发现的所有内容,没有一个与我想做的相匹配......这是 GIT 的基本情况。 为了使我的情况更简单,我有两个分支: - master(主要分支) -
我的项目: 我正在开发一辆带有 3 轴加速度计和陀螺仪的槽车,试图估计汽车姿态(x、y、z、偏航、俯仰),但我的振动噪音有一个大问题(而汽车在例如,加速度计的噪声值在 ±4[g](其中 g = 9.8
我正在尝试实现 2D Perlin 噪声来创建类似 Minecraft 的地形(Minecraft 实际上并不使用 2D Perlin 噪声)而没有悬垂或洞穴之类的东西。 我这样做的方式是创建一个 [
我有一个代表图像的二维数组。我必须向图像添加 RMS 2 单位的背景高斯噪声。我不熟悉噪声的 RMS 测量以及如何添加它。您能否告诉我如何执行此操作? 最佳答案 按照我的理解,您想要在每个像素处添加遵
更新:正式问题列表: 2D 噪声实现的浮点值(输入参数和输出)代表什么? - 部分回答,输入是坐标。输出怎么样?另外,我可以使用我的整数作为坐标的 float 吗? 1.0、122.0 等? 在 2D
我正在尝试使用 Perlin 噪声生成地形。我了解如何使用笛卡尔坐标生成它,但无法完全理解它在球体上的工作方式。我知道您可以将 2D 表面投影到球体上,但失真不会扰乱噪声分布吗?要在球体表面生成均匀噪
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我正在尝试将 fBm 实现到行星的球体上。为了创建我的球体,我将其从立方体转换为此类。不幸的是,生成的 fBm 显示为镜像补丁。此外,它只在 2 个面上执行(包装其他面的值)。当呈现为 sphere
我想知道为什么在 Simplex 推出后,Perlin 噪声至今仍然如此流行。单纯形噪声是由 Ken Perlin 自己制作的,它应该取代他的旧算法,该算法对于更高的维度来说速度较慢,但质量更好(
我终于设法为 Love 2D 编写了 Perlin 改进噪音的工作 Lua 版本。但是,当我运行它时,我得到了这个: 我想这很好。但我想要看起来更像这样的东西: 我怎样才能做到这一点? 最佳答案 第一
我终于设法为 Love 2D 编写了 Perlin 改进噪音的工作 Lua 版本。但是,当我运行它时,我得到了这个: 我想这很好。但我想要看起来更像这样的东西: 我怎样才能做到这一点? 最佳答案 第一
我正在尝试处理我从采样源数据行(Java Sound API)获得的字节数组。如果我将字节数组与小数相乘,播放流时会产生噪音。 在播放声音之前,我将立体声 wav 文件分成左右声道。这很好用。但是,如
对于我正在进行的元胞自动机项目,我需要使用不同的算法和技术随机生成二维 boolean 数组。目前,我在应用程序中只有一种随机化类型——循环遍历数组中的每个单元格并生成一个随机 double 变量,然
我叫 Chris,正在开发我的第一个 Java 游戏。到目前为止,我已经创建了一个基于图 block 的 2D 游戏,但是我的关卡是以这样一种方式完成的,如果我创建一个图像并且它全是绿色,那么绿色就代
block 之间的平滑 所以我一直在开发一个统一的游戏,想将我的世界从 150x150 的 map 扩展到一个看似无限的程序世界。我的计划是以Perlin Noise为基础,使用0-1的不同值来判断地
(此程序的依赖项:vector --any 和 JuicyPixels >= 2 。代码可用作 Gist。) {-# LANGUAGE Haskell2010 #-} {-# LANGUAGE Ban
我最近用 C# 编写了 Diamond-Square 过程生成算法的实现。但是,生成的噪声在所处理的“正方形”之间具有非常明显的边界。伪代码看起来像这样 gen() { This takes
我已经尝试了所有方法并阅读了我在互联网上看到的关于 Perlin Noise 或 Simplex Noise 的每一个链接,甚至剖析了一些我认为工作正常的 Javascript 示例。 但我仍然得到看
我在任何方面都不精通视频压缩,但目前正在从事一个使用 H.264 压缩残差图像的项目 我的问题更多是关于视频编码器的一般性问题。据我了解(正如维基百科所解释的那样), block 运动补偿编码器将当前
我是一名优秀的程序员,十分优秀!