粒子系统实现与原理

粒子系统简介:

粒子系统是指计算机图形学中模拟特定现象的技术,它在模仿自然现象、物理现象及空间扭曲上具备得天独厚的优势,能为我们实现一些真实自然而又带有随机性的效果(如爆炸、烟花、水流)提供了方便。Cocos2d-x引擎中就为我们提供了强大的粒子系统(确实Billboard Particle 是参考的cocos的源代码)。
cocos粒子系统源代码

粒子系统的分类:

现代的 Particle 有三种
• Billboard Particle
• Mesh Particle
• Ribbon Particle

在这里插入图片描述

作者本人先实现的是第一种 Billboard Particle粒子,后面还会实现 Mesh Particle。其实这两种斌没有本质上的区别,都是mesh+实例化+内存粒子+粒子的运动变换(可以各种物理量模拟仿真,甚至实现对象群)的管理等等,但是Billboard Particle粒子这种是最原始最简单的粒子,可以直接一张纹理或者播放序列帧,额Mesh Particle更通用,首先理论上可以是任意模型,而Billboard Particle粒子只是公告板或者通俗的说是四边形。再者发射的Mesh的材质可以任意,比Billboard Particle粒子效果更美更灵活~
相较而言,UE的粒子更多,还有另外三种:
在这里插入图片描述

粒子的属性:

粒子的属性包括很多,也包括各种受力,物理量的粒子。比如最常见的基本, 粒子属性: 坐标,初速度大小、速度的方向,大小,颜色,生命周期、质量、粒子的纹理(可以作成序列帧)等等。同时包括随机量!这个很关键 随机范围为 [start - variance/2, start + variance/2]-[end - variance/2, end + variance/2];

粒子发射器:

获取到所有的粒子,掌管整个粒子系统中该发射器发射的粒子,包括生命周期、消亡等等!除此之前粒子发射器有三个作用:
– 具体规定粒子产生的规则
– 具体规定粒子模拟的逻辑
– 描述如何渲染粒子(特别是透明混合等等的问题)

发射器属性:
发射器也有很多的属性,包括发射器的位置,比如我在项目中经常用的区域发射器,就有区域坐标属性,在该区域内随机new 粒子cell,甚至还可以做一个镂空,所以需要镂空坐标。对于线性发射器属性包括start的坐标,还有end的坐标

粒子的运动物理量-力场:

除此之外粒子还要受力场的影响
除此之外粒子还要受力场的影响,常见的三种力场:
在这里插入图片描述

本人身为物理课代表大学后特别工作以来物理有一些还给物理老师,所以复习啦一下下面理出各种力,匀变速运动,圆周运动等等的公式
1、 最常见的重力是一种全局设置,适用于场景中的所有物理系统。G = g*m;其中m为质量,g为重力加速度,G为重力。
2、空气阻力特别要指出,空气阻力跟速度的平方程正比,而且!空气阻力方向与运动方向或者说速度方向相反!
在这里插入图片描述

3、引力N/R^2 引力的大小N与所粒子当前与引力中心点的距离R,根据欧拉公式获取sqrt((x1-x2)2+((y1-y2)2+(z1-z2)^2)

匀变速运动公式:
速度:v1 = v0+at 其中v0初速度,a为加速度,t通常为当前一帧的时间。
位置公式 S=v0
t + (1/2 *a *t)
平均速度:V =( v1 +v2)/2
牛二定律:F=am 其中a为加速度,m为粒子的质量

圆周运动公式

1、v(线速度)=S/t=2πr/T=ωr=2πrf (S代表弧长,t代表时间,r代表半径) 。
2、ω(角速度)=θ/t=2π/T=2πn (θ表示角度或者弧度) 。   3、T(周期)=2πr/v=2π/ω 。
4、n(转速)=1/T=v/2πr=ω/2π 。
5、Fn(向心力)=mrω2=mv2/r=mr4π2/T2=mr4π2f2 。
6、an(向心加速度)=rω2=v2/r=r4π2/T2=r4π2n2 。   
7、vmax(过最高点时的最小速度)=√gr (无杆支撑)。。

粒子系统的通用核心实现:

这里主要是抛砖引玉,粒子更关注的是位置、与生命周期管理以及内存性能等等问题!至于渲染都是通用的就比如粒子也可以使用PBR,或者各种特效比如溶解,自发光、Glitter等等都是通用的这里就不特地在重复。

生命周期管理:

GLuint nr_particles = 500;             //预设置粒子的总数*每个粒子消耗内存数量 = 总数
std::vector<Particle> particles;       //粒子数组 方便管理粒子
for (GLuint i = 0; i < nr_particles; ++i) 
    particles.push_back(Particle());



GLuint nr_new_particles = 2; 
// Add new particles
for (GLuint i = 0; i < nr_new_particles; ++i) 
    {
         int unusedParticle = FirstUnusedParticle();
         RespawnParticle(particles[unusedParticle], object, offset); 
    } 
// Update all particles for 
(GLuint i = 0; i < nr_particles; ++i) 
    { 
        Particle &p = particles[i]; p.Life -= dt; 
        // reduce life 
        if (p.Life > 0.0f) { 
            // particle is alive, thus update 
            p.Position -= p.Velocity * dt;
            p.Color.a -= dt * 2.5; 
    }
 }

//但是为了提高利用率,找到第一个消亡的粒子然后用一个新产生的粒子来更新它
GLuint lastUsedParticle = 0;
    GLuint FirstUnusedParticle()
        {
            // Search from last used particle, this will usually return almost instantly  for
            (GLuint i = lastUsedParticle; i < nr_particles; ++i)
                {
                    if (particles[i].Life <= 0.0f)
                        {
                            lastUsedParticle = i;
                             return i;
                    }
    }
            // Otherwise, do a linear search for
     (GLuint i = 0; i < lastUsedParticle; ++i)
            {
                if (particles[i].Life <= 0.0f)
                    { lastUsedParticle = i; return i;
                }
            }
     // Override first particle if all others are alive
     lastUsedParticle = 0;
     return 0;
    }

粒子的渲染顶点着色器:

帧粒子或者 Billboard Particle 的vs:

#version 330 core 
layout (location = 0) 
in vec4 vertex;     // <vec2 position, vec2 texCoords> 
out vec2 TexCoords; 
out vec4 ParticleColor; 
uniform mat4 projection; 
uniform vec2 offset; 
uniform vec4 color;
     void main() { 
                    float scale = 10.0f; 
                    TexCoords = vertex.zw;
                    ParticleColor = color; 
                    gl_Position = projection * vec4((vertex.xy * scale) + offset, 0.0, 1.0); 
                    }

粒子的渲染片元着色器:

最简单的帧粒子或者 Billboard Particle 的实现ps:

#version 330 core 
in vec2 TexCoords; 
in vec4 ParticleColor; 
out vec4 color; 
uniform sampler2D sprite; 
    void main() {
                     color = (texture(sprite, TexCoords) * ParticleColor);
     }

粒子材质的设置:

particleRender:getStateBlock():setDepthTest( true );    --始终开着
particleRender:getStateBlock():setDepthWrite( );        --可以选择true 或者 false来设置是否深度写入。需要粒子的遮挡关系:粒子最后渲染队列中渲染,开启混合,深度测试始终打开,深度写入关闭,片元 将无法通过深度测试。
particleRender:getStateBlock():setBlend( true );        --公告板的粒子要开启混合,粒子的渲染顺序要当做半透明的方式排序

特殊粒子的几种实现:

1、point sprites实现拖尾的粒子

在filament中的案例有point_sprites 效果如图:

在这里插入图片描述

//拖尾的核心

   	static Vertex kVertices[NUM_POINTS];
    static float kPointSizes[NUM_POINTS];
    static uint16_t kIndices[NUM_POINTS];
    constexpr float dtheta = M_PI * 2 / NUM_POINTS;
    constexpr float dsize = MAX_POINT_SIZE / NUM_POINTS;
    constexpr float dcolor = 256.0f / NUM_POINTS;
    for (int i = 0; i < NUM_POINTS; i++) {
        const float theta = dtheta * i;
        const uint32_t c = dcolor * i;
        kVertices[i].position.x = cos(theta);
        kVertices[i].position.y = sin(theta);
        kVertices[i].color = 0xff000000u | c | (c << 8u) | (c << 16u);
        kPointSizes[i] = MIN_POINT_SIZE + dsize * i;
        kIndices[i] = i;
    }

2、追随粒子

在这里插入图片描述

本人最近实现了一种追随的粒子,跟随着某一点运动而运动,运动的轨迹是曲线的。
实现核心原理:每次update,识别点的pos 减去粒子当前的位置pos1得到的向量修改粒子的速度的方向(发射器或者粒子初始位置任意),速度大小恒定不变。

3、点精灵(Point Sprites)实现粒子系统:

所谓的点精灵在OpenGL中其实是图元使用了 GL_POINTS类型来回只一个顶点。当然这个GL_POINTS类型的顶点不是一个像素,而是屏幕空间中一个正方形,可以调解大小;点精灵的大小有两种方法绘制,一种是在应用程序CPU中调用函数glPointSize()设置点的大小,这种方式点的大小是固定的,不会随着相机的离远而变小。还有一种是在顶点着色器中设置,使用内置的输出变两gl_PointSize来设置,可以根据它们相对于近平面的距离来进行一个线性映射。

对于点精灵(Point Sprites)的材质也是类似的最简单可以使用纯色,,然后使用gl_PointCoord与discard控制渲染范围或纹理的采样贴到点精灵上等等:

//片元着色器
#version 330 core
uniform sampler2D sprite_texture;
unifrom vec4 color1;
unifrom vec4 color2;
out vec4 FragColor;
void main()
{
    vec2 temp = gl_PointCoord - vec2(0.5);
    float f = dot(temp, temp);
    
    if(f > 0.25)
    {
        discard;
    }
    FragColor = mix(color1, color2, smoothstep(0.1, 0.25, f));
};

点精灵(Point Sprites)和公告板(Billboards)对比:
点精灵(Point Sprites)的优点:

  1. 性能更优:使用点精灵可以减少CPU到GPU的数据传输,提高渲染性能。因为每个点精灵只需要一个顶点和一个颜色或纹理坐标,而公告板需要四个顶点和四个颜色或纹理坐标。
  2. 简单易用:点精灵在制作粒子系统时更简单,编码也更加直观。

点精灵(Point Sprites)的缺点:

  1. 缺乏灵活性:所有的点精灵都是正方形并且总是面向摄像机,这限制了它们的使用场景。
  2. 不支持深度排序:点精灵不适合于需要深度排序,比如半透明的粒子。

公告板(Billboards)的优点:

  1. 高度可定制:公告板通常使用四边形表示,可以通过调整四个顶点的位置来创建任意形状和大小的粒子。
  2. 支持深度排序:公告板允许对粒子进行深度排序,所以它们可以正确地处理半透明的粒子。

公告板(Billboards)的缺点:

  1. 性能较差:与点精灵相比,公告板需要传输更多的数据(两个三角形与四个顶点),因此需要更多的CPU和GPU资源。
  2. 更复杂:需要手动计算四个顶点的位置来确保公告板面向摄像机,比使用点精灵更为复杂。

4、基于GPU的粒子

上述的两种都是基于CPU的粒子,因为它们都是在GPU中进行运算的,其实对于大量的顶点或者变换还是GPU最擅长,而且能充分发挥GPU的性能!感兴趣的可以详细看一下

在这里插入图片描述
github地址:keijiro/Skinner: Special Effects with Skinned Mesh in Unity (github.com)

Skinner is a collection of special effects that use vertices of an
animating skinned mesh as emitting points. It uses a special
replacement shader to convert vertex positions into GPU-friendly data,
and thus it avoids spending extra memory and CPU time for handling
them (uses GPU resources instead).

  • Skinner 是一种特效,它使用动画技术中的蒙皮网格的顶点作为发射点。 它使用特殊的替换着色器将顶点位置转换为GPU能识别并且使用的数据,从而避免花费额外的内存和CPU时间来处理它们(而是使用GPU资源)。

粒子系统的性能优化:

CPU的粒子:

GPU的粒子:

  • 高并行运算,适合大量粒子的模拟计算
  • 可以释放CPU功耗来进行游戏本身计算
  • 方便获得深度缓冲来做遮挡判断

参考资料:

Skinned Mesh原理解析和一个最简单的实现示例_n5的博客-CSDN博客
结合源码看《我所理解的cocos2dx-3.0》—— 粒子系统_fztfztfzt的博客-CSDN博客
GAMES104_Lecture12_Effects.pdf (boomingtech.com)

  • 6
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
黄小平编著的《粒子滤波原理及应用》——Matlab仿真书中代码。本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握本书主要介绍粒 子滤波的基原理及其在非线性系统中应用。为方便读者快速掌握子滤波的精髓,本 书采用原理介绍 书采用原理介绍 +实例应用 +MATLAB +MATLAB +MATLAB 程序仿真 +中文注释相结合的方式, 中文注释相结合的方式, 向读者介绍滤波的原理实现过程。 向读者介绍滤波的原理实现过程。 向读者介绍滤波的原理实现过程。 向读者介绍滤波的原理实现过程。 向读者介绍滤波的原理实现过程。 向读者介绍滤波的原理实现过程。 向读者介绍滤波的原理实现过程。 本书共 本书共 9章,第 章,第 1章绪论,介绍粒子滤波的发 章绪论,介绍粒子滤波的发 章绪论,介绍粒子滤波的发 章绪论,介绍粒子滤波的发 章绪论,介绍粒子滤波的发 章绪论,介绍粒子滤波的发 展状况; 展状况; 第 2章简略地介绍 章简略地介绍 章简略地介绍 MATLAB MATLAB 算法仿真编程基础,便于零 算法仿真编程基础,便于零 算法仿真编程基础,便于零 算法仿真编程基础,便于零 算法仿真编程基础,便于零 算法仿真编程基础,便于零 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 基础的读者学习后续章节介绍原理; 第 3章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 章介绍与粒子滤波相关的概率论基础;第 4章介绍蒙特卡洛的基本原 章介绍蒙特卡洛的基本原 章介绍蒙特卡洛的基本原 章介绍蒙特卡洛的基本原 章介绍蒙特卡洛的基本原 章介绍蒙特卡洛的基本原 理;第 理;第 理;第 5章介绍粒 章介绍粒 子滤波的基本原理;第 子滤波的基本原理;第 子滤波的基本原理;第 子滤波的基本原理;第 子滤波的基本原理;第 6章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 章介绍粒子滤波的改进算法,主要是 EPF EPF算法和 UPF 算法。第 算法。第 7章和第 8章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 章为粒子滤波在目标跟踪、电池参数估计中的应用;第 9章为 SimulinkSimulink Simulink SimulinkSimulink 环境下粒子 环境下粒子 环境下粒子 滤波器的设计。 滤波器的设计。 滤波器的设计。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值