https://www.bbsmax.com/A/n2d9P1Q5Dv/

刚刚结束完地球切片的渲染调度后,打算介绍一下目前大家都很关注的3D Tiles方面的内容,但发现要讲3D Tiles,或者充分理解它,需要对DataSource,Primitive要有基础,而这要求对最底层的渲染模块,也就是Render有一个了解,真的是书到用时方恨少的代入感,索性从头到尾的说清楚。所以,下面几篇(估算四篇)先把Render模块介绍清楚。

Render,顾名思义就是渲染模块,在Cesium中,所有的对象,都是通过Render模块完成最终调用WebGL的渲染过程。换句话说,Render是Cesium的渲染引擎模块,在Cesium的Geometry和WebGL之间搭建了一个桥梁。至于为什么不直接调用WebGL接口,还需要封装一套,这样做有何意义呢,这就是一个比较发散的问题了。首先WebGL提供的是函数,而在真正的渲染中,会有很多状态的切换,而且随着逻辑的负责而难以维护,而面向对象的思想的精髓就在于对对象状态的管理,这也是为什么很多三维应用都提供了自己的渲染引擎模块,个人觉得主要的价值在于易用性和状态的管理。

首先,目前主流的浏览器基本支持WebGL 1.0的标准,对应的是OpenGL ES2.0,也就是可编程渲染管线,采用GPU渲染,性能上有保证。据我了解,只有FireFox的一些实验版本支持WebGL2.0,也就是OpenGL ES3.0的标准。如果对WebGL的基本接口还不熟悉,推荐这本《WebGL编程指南》,除了感觉价格略贵外,还是一本很不错的入门书。我们的重点是学习Cesium如何设计并封装WebGL接口,并不会过多纠结在WebGL本身的理解,所以假设你对WebGL有一个基本的了解。

Buffer

任意一个非参数化的几何对象(参数化的可以通过差值转为非参数化,比如一个参数化的圆,对应的是圆心和半径,我们通过差值,将该圆转为一个多边形,圆周对应的点越密,则效果越好,而代价也是点数据越大),对应的就是N个顶点(Node)之间的空间关系。当我们想要通过WebGL渲染该几何对象时,首先就是要讲该几何对象转化为WebGL可以识别的数据格式:1构建该对象的顶点数组,里面包括每一个点的XYZ位置(必须),该点的颜色,纹理坐标,法线等信息;2构建该对象对应的索引信息,也就是点之间的先后顺序。

上述说的就是一个VBO(顶点缓存)的概念,WebGL提供了bufferData接口,我们对其中的参数做一个简单介绍:

  1. void gl.bufferData(target, size, usage);
  2. void gl.bufferData(target, data, usage);
  • target

    gl.ARRAY_BUFFER 该数据为顶点数据,比如位置信息,颜色,法线,纹理坐标或自定义的一些属性信息

    gl.ELEMENT_ARRAY_BUFFER 该数据为顶点索引

  • size

    数据长度,此时只是预留了对应长度的空间,里面的数据为空

  • data

    数据内容,此时WebGL内部可以判断该数据内容对应的长度

  • usage

    gl.STATIC_DRAW 静态数据,数据保存在GPU中,适合一次写入不再更改,多次使用情况

    gl.DYNAMIC_DRAW 动态数据,保存在内存中,适合多次写入,多次使用下调用情况

    gl.STREAM_DRAW 流数据,保存在GPU中,适合一次写入一次使用的情况

如下是直接调用WebGL提供的方法创建顶点属性的代码:

  1. // 获取WebGL对象
  2. var canvas = document.getElementById("canvas");
  3. var gl = canvas.getContext("webgl");
  4. // 创建顶点缓存
  5. var buffer = gl.createBuffer();
  6. // 绑定该顶点缓存类型为顶点属性数据
  7. gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
  8. // 指定其对应的数据长度及方式
  9. gl.bufferData(gl.ARRAY_BUFFER, 1024, gl.STATIC_DRAW);
  10. // 解除绑定
  11. gl.bindBuffer(bufferTarget, null);

通过上面的介绍,可见顶点属性和顶点索引的调用方法完全相同,逻辑上的创建过程也如出一辙,只是具体的参数稍有不同,因此,在Cesium中把创建VBO的过程化的函数封装为一个抽象的Buffer类,伪代码如下:

  1. function Buffer(options) {
  2. var gl = options.context._gl;
  3. var bufferTarget = options.bufferTarget;
  4. var typedArray = options.typedArray;
  5. var sizeInBytes = options.sizeInBytes;
  6. var usage = options.usage;
  7. var hasArray = defined(typedArray);
  8. if (hasArray) {
  9. sizeInBytes = typedArray.byteLength;
  10. }
  11. // ……
  12. var buffer = gl.createBuffer();
  13. gl.bindBuffer(bufferTarget, buffer);
  14. gl.bufferData(bufferTarget, hasArray ? typedArray : sizeInBytes, usage);
  15. gl.bindBuffer(bufferTarget, null);
  16. // ……
  17. this._gl = gl;
  18. this._bufferTarget = bufferTarget;
  19. this._sizeInBytes = sizeInBytes;
  20. this._usage = usage;
  21. this._buffer = buffer;
  22. this.vertexArrayDestroyable = true;
  23. }

可见,这个过程和刚才WebGL调用的方式几乎一模一样,只是把所需要的参数都封装了一下,并将调用函数的返回值,作为属性保存在该Buffer对象中。于是,一个过程式的函数调用封装成了一个Ojbect,即可以重用,也方便内部细节的管理。

当然,如果只有如上的一个方法也能用,但输入的参数不太少,只是一个很简单的函数封装而已,使用上并没有太多简化,而且也不方便理解,只能算是一个半成品。接着Cesium在Buffer类中提供了Buffer.createVertexBuffer和Buffer.createIndexBuffer方法,算是基于Buffer的一个二次加工:

  1. Buffer.createVertexBuffer = function(options) {
  2. return new Buffer({
  3. context: options.context,
  4. bufferTarget: WebGLConstants.ARRAY_BUFFER,
  5. typedArray: options.typedArray,
  6. sizeInBytes: options.sizeInBytes,
  7. usage: options.usage
  8. });
  9. };
  10. Buffer.createIndexBuffer = function(options) {
  11. return new Buffer({
  12. context: options.context,
  13. bufferTarget : WebGLConstants.ELEMENT_ARRAY_BUFFER,
  14. typedArray : options.typedArray,
  15. sizeInBytes : options.sizeInBytes,
  16. usage : options.usage
  17. });
  18. };

这下用起来就so easy了,用户创建顶点属性和顶点索引的范例如下:

  1. var buffer = Buffer.createVertexBuffer({
  2. context : context,
  3. sizeInBytes : 16,
  4. usage : BufferUsage.DYNAMIC_DRAW
  5. });
  6. var buffer = Buffer.createIndexBuffer({
  7. context : context,
  8. typedArray : new Uint16Array([0, 1, 2]),
  9. usage : BufferUsage.STATIC_DRAW,
  10. indexDatatype : IndexDatatype.UNSIGNED_SHORT
  11. });

当然,如果你此时是用的DYNAMIC_DRAW的方式,并没有指定bufferdata,则创建该顶点缓存对象,但数据还是空的,在获取数据后可以调用Buffer.prototype.copyFromArrayView方法更新一下该数据。可见,Buffer类的设计还是比较周全的,考虑到数据更新可能存在这种不确定的逻辑:

  1. Buffer.prototype.copyFromArrayView = function(arrayView, offsetInBytes) {
  2. var gl = this._gl;
  3. var target = this._bufferTarget;
  4. gl.bindBuffer(target, this._buffer);
  5. gl.bufferSubData(target, offsetInBytes, arrayView);
  6. gl.bindBuffer(target, null);
  7. };

如上就是创建VBO的一个完整过程,可以仔细对比一下两者之间的不同。因为VBO是渲染中最基本的,也是最重要的概念和渲染对象,通过Buffer类对这个过程进行分装和管理,虽然难度不大,但意义却很重要。

通过Buffer,我们可以将Primitive(图元)中的几何数据转化为VBO,这相当于创建了该几何对象的骨架。通常我们还需要纹理信息,贴在这个骨架的表面,让它看上去有血有肉,惟妙惟肖。下一篇我们介绍Cesium是如何封装Texture。

Cesium原理篇:6 Renderer模块(1: Buffer)【转】的更多相关文章

  1. Cesium原理篇:7最长的一帧之Entity(下)

    上一篇,我们介绍了当我们添加一个Entity时,通过Graphics封装其对应参数,通过EntityCollection.Add方法,将EntityCollection的Entity传递到DataSo ...

  2. Cesium原理篇:5最长的一帧之影像

    如果把地球比做一个人,地形就相当于这个人的骨骼,而影像就相当于这个人的外表了.之前的几个系列,我们全面的介绍了Cesium的地形内容,详见: Cesium原理篇:1最长的一帧之渲染调度 Cesium原 ...

  3. Cesium原理篇:3最长的一帧之地形(2:高度图)

           这一篇,接着上一篇,内容集中在高度图方式构建地球网格的细节方面.        此时,Globe对每一个切片(GlobeSurfaceTile)创建对应的TileTerrain类,用来维 ...

  4. Cesium原理篇:6 Renderer模块(1: Buffer)

    刚刚结束完地球切片的渲染调度后,打算介绍一下目前大家都很关注的3D Tiles方面的内容,但发现要讲3D Tiles,或者充分理解它,需要对DataSource,Primitive要有基础,而这要求对 ...

  5. Cesium原理篇:6 Renderer模块(2: Texture)

    Texture也是WebGL中重要的概念,使用起来也很简单.但有句话叫大道至简,如果真的想要用好纹理,里面的水其实也是很深的.下面我们来一探究竟. 下面是WebGL中创建一个纹理的最简过程: var ...

  6. Cesium原理篇:6 Render模块(6: Instance实例化)

    最近研究Cesium的实例化,尽管该技术需要在WebGL2.0,也就是OpenGL ES3.0才支持.调试源码的时候眼前一亮,发现VAO和glDrawBuffers都不是WebGL1.0的标准函数,都 ...

  7. Cesium原理篇:6 Render模块(6: Instance实例化)【转】

    https://www.cnblogs.com/fuckgiser/p/6027520.html 最近研究Cesium的实例化,尽管该技术需要在WebGL2.0,也就是OpenGL ES3.0才支持. ...

  8. Cesium原理篇:6 Render模块(3: Shader)

    在介绍Renderer的第一篇,我就提到WebGL1.0对应的是OpenGL ES2.0,也就是可编程渲染管线.之所以单独强调这一点,算是为本篇埋下一个伏笔.通过前两篇,我们介绍了VBO和Textur ...

  9. Cesium原理篇:6 Render模块(5: VAO&RenderState&Command)

    VAO VAO(Vertext Array Object),中文是顶点数组对象.之前在<Buffer>一文中,我们介绍了Cesium如何创建VBO的过程,而VAO可以简单的认为是基于VBO ...

随机推荐

  1. expect脚本远程登录、远程执行命令和脚本传参简单用法

    expect介绍: 最近想写一个自动化安装脚本,涉及到远程登录.分发文件包.远程执行命令等,其中少不了来回输入登录密码,交互式输入命令等,这样就大大降低了效率,那么有什么方法能解决呢?不妨试试expe ...

  2. Host is not allowed to connect to this MySQL server

    解决方法: [root@GYQ-Prod-Zabbix ~]# mysql -u root -p Enter password: Welcome to the MariaDB monitor. Com ...

  3. mysql遇到时区问题的坑(Java解决方案)

    最近项目遇到一个坑,就是server和db之间存在时区问题,本人的db是utc时间, 可以使用代码设置时区来解决,本人这里使用joda三方包,joda蛮好用的,具体用法这里不做详细描述. 先引入pom ...

  4. C# 6.0 中的新增功能(.NET Framework 4.6 与 Visual Studio 2015 )

    C#6.0 在 2015 年7月随着.NET Framework 4.6 一同发布,后期发布了.NET Framework 4.6.1,4.6.2. 一.自动属性初始化(Auto-property i ...

  5. 【测试工具】moco入门(一)

    转自:https://www.cnblogs.com/tangqiu/p/9493147.html 简单来说,Moco就是解决了开发前端时没有后端支持,开发接口时依赖没有到位的尴尬场景.当然Moco的 ...

  6. 应该知道的linux命令

    常用命令 1.在compose Bar下可以对多个服务器同时进行操作.选择To All Sessions 2. 查看JAVA进程: ps -ef | grep java ps auxf | grep ...

  7. Vue 项目环境搭建

    Vue项目环境搭建 ''' 1) 安装node 官网下载安装包,傻瓜式安装:https://nodejs.org/zh-cn/ 2) 换源安装cnpm >: npm install -g cnp ...

  8. dosbox+masm5.0编译汇编文件

    在去年写过如何bc3.1编译ucos,不过现在很少去用到,但是那是用dosbox也是懵懵懂懂的,参见https://blog.csdn.net/liming0931/article/details/8 ...

  9. C#延迟初始化Lazy<T>

    1. 概述 我们创建某一个对象需要很大的消耗,而这个对象在运行过程中又不一定用到,为了避免每次运行都创建该对象,这时候延迟初始化(也叫延迟实例化)就出场了. 延迟初始化出现于.NET 4.0,主要用于 ...

  10. CQOI2016 不同的最小割 (最小割树模板)(等价流树的Gusfield构造算法)

    题目 最小割树模板 算法详解及证明见: 2016年国家队候选队员论文 <浅谈无向图最小割问题的一些算法及应用--绍兴一中 王文涛> 3.2节 CODE #include <bits/ ...