three.js 粒子效果(分别基于 CPU & GPU 实现)

前段时间做了一个基于 CPU 和 GPU 对比的粒子效果丢在学习 WebGL 的 RTX 群里,技术上没有多作讲解,有同学反馈看不太懂 GPU 版本,干脆开一篇文章,重点讲解基于 GPU 开发的版本。

一、概况

废话不多说,先丢上demo,用移动设备更能明显感觉性能差异。

维护粒子位移、颜色、尺寸

维护粒子位移

结论:

同时需要维护多种粒子特征变化时,GPU有明显优势。 只是维护粒子位移时,GPU版本稍流畅,但优势并不明显。 当然,这还得具体到设备,一些中低端Android机器,GPU太渣,不如CPU计算。

二、技术实现

three.js中,粒子效果的实现方式大概分为三种:

1、Javascript直接计算粒子的状态变化,即基于CPU实现; 2、Javascript通知顶点着色器粒子的生命周期,由顶点着色器运行,即基于GPU实现; 3、粒子生成与状态维护全部由片元着色器负责,即屏幕特效,同样是基于GPU中实现。 第3种方式本文暂不介绍。

2.1、基于CPU实现

护位移、颜色、尺寸: http://tgideas.qq.com/2017/three/shader/particle-gpu/cpu.html

维护位移: http://tgideas.qq.com/2017/three/shader/particle-gpu/gpu-position.html

步骤1&2:

首先加载由三维软件制作的几何体,然后生成粒子系统 。

var material = new THREE.PointsMaterial({size:4, color:0xff0000});
var particleSystem = new THREE.Points(geometry , material);

从代码中可以看出,材质是针对整介粒子系统设置的,所以只能维护粒子位移。

如果要维护粒子颜色、尺寸呢?

我们必须为每个粒子设置不同的材质,由此也造成不小的性能损耗 。

步骤3:

使用Tween修改所有顶点位置。

tween = new TWEEN.Tween(pos).to({val: 0}, 2000).easing(TWEEN.Easing.Quadratic.InOut).delay(1000).onUpdate(callback);
function callback(){
    var val = this.val;
    var particles = particleSystem.geometry.vertices;
    for(var i = 0; i < particles.length; i++) {
        var pos = particles[i];
        pos.x = position1[i].x * val + position2[i].x * (1-val);
        pos.y = position1[i].y * val + position2[i].y * (1-val);
        pos.z = position1[i].z * val + position2[i].z * (1-val);
    }
    particleSystem.geometry.verticesNeedUpdate = true;
}

从代码中可以看出,粒子的状态都是通过Javascript,由CPU来计算。

2.2、基于GPU实现

维护粒子位移、颜色、尺寸: http://tgideas.qq.com/2017/three/shader/particle-gpu/gpu.html

对比CPU实现流程图,我们会发现,Tween并不直接计算所有顶点位置,而是只通知动画运行时间,由顶点着色器来完成具体运算。

既然运算部分在顶点着色器,那么,需要我们自己书写着色器(opengl es),所以我们选用three.js中的ShaderMaterial。

步骤1:

首先生成粒子系统:

var uniforms = {
    texture:{value: new THREE.TextureLoader().load( "dot.png")},
    val: {value: 1.0}
};
var shaderMaterial = new THREE.ShaderMaterial({
    uniforms:     uniforms,
    vertexShader:   document.getElementById('vertexshader').textContent,
    fragmentShader: document.getElementById('fragmentshader').textContent,
    blending:       THREE.AdditiveBlending,
    depthTest:      false,
    transparent:    true
});
particleSystem = new THREE.Points(moreObj, shaderMaterial);

uniforms是连接javascript与着色器的通道。

uniforms.val 即由tween来维护的动画运行值。

vertexShader和fragmentShader,即我们要定义的顶点着色器,和片元着色器,它们负责具体的粒子状态的运算,我们定义在网页中。

步骤2:

定义顶点着色器:

attribute float size; // 粒子尺寸
attribute vec3 position2; // 目标顶点位置
uniform float val; // 动画运行时间
varying vec3 vPos; // 将顶点位置传输给片元着色器

void main() {
    // 计算粒子位置
    vPos.x = position.x * val + position2.x * (1.-val);
    vPos.y = position.y* val + position2.y * (1.-val);
    vPos.z = position.z* val + position2.z * (1.-val);
    // 坐标转换
    vec4 mvPosition = modelViewMatrix * vec4( vPos, 1.0 );
    gl_PointSize = size * ( 300.0 / -mvPosition.z );
    gl_Position = projectionMatrix * mvPosition;

}

three.js内置,自动传递给顶点着色器的变量: attribute position - 顶点坐标 mat4 modelViewMatrix - 模型+视图矩阵 mat4 projectionMatrix - 投影矩阵

定义片元着色器:

uniform sampler2D texture;
varying vec3 vPos;

void main() {
    // 计算粒子颜色,通过位置
    vec3 vColor = vec3(1.0, 0., 0.);
    vColor.r = vPos.z/50.;
    vColor.g = vPos.y/50.;
    vColor.b = vPos.x/50.;

    gl_FragColor = vec4(vColor, 1.0 );
    // 顶点贴图
    gl_FragColor = gl_FragColor * texture2D( texture, gl_PointCoord );

}

步骤3:

负责维护粒子运行时间:

tween = new TWEEN.Tween(pos).to({val: 0}, 2000).onUpdate(callback);
function callback(){
    particleSystem.material.uniforms.val.value = this.val;
}

三、延伸阅读

  • 类THREE.Points做了什么?

其实真没干什么,主要是申明它的type是Points。 当我们执行渲染时,WebGL会绘制Point,即调用gl.drawArrays(gl.POINTS… 而通常,比如type为Mesh时,three.js会调用gl.drawArrays(gl.TRIANGLES…

  • 类THREE.PointsMaterial做了什么?

同样,点材质也是three.js最简单的类之一,相对于基类Material,它多做的事情只是传递了size,即点的尺寸这个值。

原创声明,本文系作者授权云+社区发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

编辑于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏程序人生

谈谈状态机

题记:上周做 BBL 里讲了我们 Tubi TV 内部做 DSL 的一些简单实践,大家反馈不错。有同事建议我给大家先补补 FSM,之后再进阶 CFG,可能会更顺...

39470
来自专栏架构师之路

洗稿,技术上怎么判断文章相似性?

这几天“差评洗稿”的事情,闹得沸沸扬扬,楼主本身也是内容创作的手艺人,简单说两句。

21340
来自专栏滕先生的博客

UIDynamic 物理引擎概念介绍UIDynamicAnimator(动画者)动力行为(UIDynamicBehavior)一、抽象类 UIDynamicBehavior二、UIGravityBeh

49770
来自专栏小樱的经验随笔

1022: [SHOI2008]小约翰的游戏John【Nim博弈,新生必做的水题】

1022: [SHOI2008]小约翰的游戏John Time Limit: 1 Sec  Memory Limit: 162 MB Submit: 2709 ...

33180
来自专栏新智元

Jeff Dean推荐:用TPU跑Julia程序,只需不到1000行代码

Julia是一门集众家所长的编程语言。随着Julia 1.0在8月初正式发布,Julia语言已然成为机器学习编程的新宠。

11510
来自专栏机器之心

资源 | 知网(HowNet)知识库的简单调用指南

机器之心整理 参与:蒋思源 机器之心曾采访过语知科技的董强先生,在那一篇文章中,我们详细讨论了基于知网知识库的 NLP 解决方案。虽然我们已经了解了这种方法的潜...

79350
来自专栏落影的专栏

OpenGL ES实践教程(九)OpenGL与视频混合

前言 前面的实践教程: OpenGL ES实践教程1-Demo01-AVPlayer OpenGL ES实践教程2-Demo02-摄像头采集数据和渲染 O...

61550
来自专栏飞总聊IT

大数据那些事(10):李逵麻子,李鬼坑人--BigTable的数据模型

今天我们回归技术路线,讲讲Google三驾马车里的BigTable。以前有个说法叫做麻子不叫麻子,叫坑人。取其原意是满脸是坑的人,谐音表示人被坑了。我们知道水浒...

41970
来自专栏CDA数据分析师

10个令人相见恨晚的R语言包

新媒体管家 ? 大约3年前我开始使用R,起初进展很慢,与我习惯的语言相比,语法更加直观也比较简单,而且需要一段时间才能习惯于细微的差别。我还不清楚语言的力量与社...

224100
来自专栏linux驱动个人学习

android的hwc浅析【转】

62060

扫码关注云+社区

领取腾讯云代金券