OpenGL 入门(一)—— OpenGL 基础

news2025/1/10 0:35:14

1、OpenGL 基础知识

1.1 OpenGL 简介

OpenGL(Open Graphics Library)是图形领域的工业标准,是一套跨编程语言、跨平台、专业的图形编程(软件)接口。它用于二维、三维图像,是一个功能强大,调用方便的底层图形库。它与硬件无关,可以在不同的平台如 Windows、Linux、Mac、Android、IOS 之间进行移植。因此,支持 OpenGL 的软件具有很好的移植性,可以获得非常广泛的应用(比如 PS 在部分功能和操作中使用 OpenGL 加速,以提高图像处理和渲染的性能)。

对于 Android 系统而言,它支持的是 OpenGL ES。OpenGL ES 是针对手机、PDA 和游戏主机等嵌入式设备设计的 OpenGL API 的子集。下面是 Android 系统版本与 OpenGL ES 的支持版本的一般对照表(具体的设备和制造商可能会在特定型号上进行适配和调整):

  • Android 4.0(Ice Cream Sandwich):OpenGL ES 2.0
  • Android 4.1 - 4.3(Jelly Bean):OpenGL ES 2.0/3.0
  • Android 4.4(KitKat):OpenGL ES 2.0/3.0/3.1
  • Android 5.0 - 5.1(Lollipop):OpenGL ES 3.1/3.2
  • Android 6.0(Marshmallow):OpenGL ES 3.1/3.2
  • Android 7.0 - 12.0:OpenGL ES 3.2

OpenGL 官方上提供了很多资料,作为 Android 开发者需要关注 OpenSL ES,可以在OpenGL 官网页面找到两个 OpenSL ES 的资料:

请添加图片描述

第一个是 OpenGL ES 的官方文档:

2024-3-18.OpenGL ES官方文档0

可以在右侧选择版本:

2024-3-18.OpenGL ES官方文档

第二个是 OpenGL ES 的 API 快速查找手册:

在这里插入图片描述

在这里插入图片描述

1.2 OpenGL 绘制原理

OpenGL 是可以绘制 3D 图像的,比如绘制一头牛的过程如下:

2024-3-18.OpenGL绘制流程

  1. Vertex Processing:首先将牛抽象为 3D 网格,在网格中牛由无数个像素点组成,将这些像素点视为顶点,每 3 个顶点可以构成一个三角形,这个三角形可以视为一个纹理,多个纹理可以组成一个面,最终形成一头牛。其中,将 3D 网格转化为顶点的过程是由顶点着色器完成的
  2. Rasterization:光栅化,将几何图形转换为在屏幕上显示的像素网格得到片元
  3. Raster Operations:进行纹理过滤,使用片元着色器对片元上色
  4. Fragment Processing:经过片元处理得到最终要输出的图像

以上过程涉及到一些 OpenGL 的专业名词,解释如下:

  1. 纹理(Texture):是将图像或图案应用到几何图形上的一种技术。
    • 纹理可以理解为在三维空间中的一个图像或图案,它可以覆盖在三维对象的表面上,给出了表面的颜色和细节。实际上是通过将二维的纹理映射到三维的几何图形的表面上,从而实现对几何图形进行贴图,使其呈现出真实的外观和细节。
    • 纹理通常是由像素组成的二维图像,可以是从图像文件加载的位图,或者是通过代码生成的程序纹理。在 OpenGL 中,纹理坐标(Texture Coordinates)被用来映射纹理到三维几何图形上的对应位置。
    • 纹理可以看作是一个二维数组,由一系列像素组成。每个像素包含了图像的颜色信息或其他数据。纹理可以包含不同的图像格式,例如 RGB、RGBA、灰度图等。
  2. 顶点着色器(Vertex Shader):顶点着色器是一种在图形渲染管线中执行的程序,用于处理输入的顶点数据。它以顶点为基本单位,对每个输入顶点进行处理和变换,例如对顶点位置、颜色、法线等进行变换、计算和插值。顶点着色器通常用于执行顶点变换、几何操作和顶点属性的计算。
  3. 光栅化(Rasterization):是计算机图形学中的一个重要概念,用于将几何图形转换为在屏幕上显示的像素图像。光栅化的过程涉及将连续的几何图形转换为离散的像素表示。在三维图形渲染中,几何图形通常是由三角形组成的三角网格(Triangle Mesh)。光栅化将每个三角形映射到屏幕上的像素,并确定每个像素的颜色和深度值。其主要步骤包括:
    • 顶点处理(Vertex Processing):在光栅化之前,对每个顶点进行变换、投影和其他处理,以将顶点从三维空间转换到屏幕空间。
    • 三角形设置(Triangle Setup):确定每个三角形的边界框(bounding box)或边界区域,以限定光栅化的范围。
    • 逐像素处理(Pixel Processing):对于每个位于三角形边界框内的像素,确定其是否位于三角形内部,并计算其颜色、纹理坐标、深度值等属性。
    • 插值计算(Interpolation):根据顶点属性的值,通过插值计算来获取每个像素的准确属性值,如颜色插值、纹理坐标插值等。
    • 像素输出(Pixel Output):根据计算得到的像素属性值,将其写入帧缓冲区(Frame Buffer),最终形成屏幕上的图像
  4. 片元着色器(Fragment Shader):片元着色器是一种在图形渲染管线中执行的程序,用于处理光栅化后的片元(像素)数据。它以片元为基本单位,对每个片元进行处理和计算,例如计算片元的颜色、法线、光照等。片元着色器通常用于执行光照模型、纹理采样、混合和其他片元级别的计算。

顶点着色器和片元着色器需要程序员自行实现。

1.3 Android 绘制原理

Android 在绘制时实际上就用到了 OpenGL。比如绘制一个 Button:

3.3.1-Android绘制原理

大致步骤如下:

  • LayoutInflater 解析布局文件,将 Button 从 xml 的标签解析成包含边界和宽高等信息的 Button 对象
  • CPU 对 Button 对象内的边界、宽高、颜色等进行计算,处理成纹理(多维的向量图形)交给 GPU
  • GPU 负责填充,将向量图形栅格化(转换成像素位图图像)再画到屏幕上

实际上 OpenGL 就是工作在 GPU 当中的,下图从 CPU、GPU 和屏幕的维度描绘了绘制过程:

3.3.1-Android渲染机制分析.PNG

既然提到 CPU 与 GPU,就要简单看一下二者的区别与联系:

3.3.1-CPU与GPU的区别1

3.3.1-CPU与GPU的区别2

1.4 OpenGL 坐标系

OpenGL 编程会常用到如下三种坐标系:

2024-3-18.OpenGL世界坐标系OpenGL世界坐标系的边界为1
Android坐标系原点是屏幕左上角
纹理坐标系原点是左下角
  • OpenGL 世界坐标系的原点 (0,0) 在中心,边界最大坐标为 1
  • Android 屏幕坐标系的原点 (0,0) 在左上角,边界也是 1
  • OpenGL 纹理坐标系的原点 (0,0) 在左下角,边界还是 1

OpenGL 世界坐标系与 Android 屏幕坐标系的顶点是有对应关系的,如图所示:

2024-4-8.OpenGL坐标系对应关系

2、OpenGL SL 语法基础

OpenGL 编程使用的着色器语言。

着色器语言(Shader Language)是一种特定于图形编程的编程语言,用于编写顶点着色器、片元着色器和其他类型的着色器程序。在 OpenGL 中,常用的着色器语言是 GLSL(OpenGL Shading Language),它是一种高级着色器语言,用于描述和执行图形渲染管线中的着色器程序。GLSL 提供了丰富的语法和内置函数,使得开发者可以编写灵活和高效的着色器代码。

以下是后续 Demo 中常用的语法内容:

数据类型描述
float浮点型
vec2含两个浮点型数据的向量
vec4含四个浮点型数据的向量(xyzw、rgba、stpq)
sampler2D2D纹理采样器(代表一层纹理,用uniform来修饰)
修饰符描述
attribute属性变量。只能用于顶点着色器中。一般用该变量来表示一些顶点数据,如:顶点坐标、纹理坐标、颜色等。
uniform一致变量。在着色器执行期间一致变量的值是不变的。与 const 常量不同的是,这个值在编译时期是未知的是由着色器外部初始化的。
varying易变变量。是从顶点着色器传递到片元着色器的数据变量。
内建函数描述
texture2D(采样器, 坐标) 采样指定位置的纹理

这里顺便了解一下 OpenGL 函数的命名格式:

<函数库前缀> <根命令> <可选的参数数量> <可选的参数类型>

2024-3-18.OpenGL命名规则

使用参数数量和参数类型是因为面向过程的 C 语言中没有函数重载。

内建变量描述
gl_Positionvec4类型,表示顶点着色器中顶点位置
gl_FragColorvec4类型,表示片元着色器中颜色
精度名称描述
precision lowp低精度
precision mediump中精度
precision highp高精度

举个示例,比如要声明一个 4 * 4 的矩阵:

mat4 m1 = mat4(
	1.0, 2.0, 3.0, 4.0,
    1.0, 2.0, 3.0, 4.0,
    1.0, 2.0, 3.0, 4.0,
    1.0, 2.0, 3.0, 4.0
)
    
mat4 m2 = mat4(2.0)

mat4 m3 = mat4(
    2.0, 0.0, 0.0, 0.0,
    0.0, 2.0, 0.0, 0.0,
    0.0, 0.0, 2.0, 0.0,
    0.0, 0.0, 0.0, 2.0
)

其中 m2 初始化的结果与 m3 相同,即初始化左上角到右下角对角线上位置上的数据,其余位置均为 0。

3、OpenGL SL 简单使用

这一节我们使用 OpenGL SL 在 Android 屏幕上画一个等腰直角三角形。

3.1 OpenGL SL 简介

首先我们要清楚,OpenGL SL 本质上是一个 NDK 的动态库,在 NDK 的 toolchains 中可以找到它们:

2024-3-26.OpenGL Native库1

但是 Google 将 NDK 接口封装为 Java API 方便上层调用:

2024-3-26.OpenGL上层封装

图形渲染管线

OpenGL 图形渲染管线(计算机图形学中用于将三维模型转换为最终渲染图像的流程和算法)的简化流程图,它展示了顶点着色器和片元着色器之间的数据流:

2024-3-18.GLSL

OpenGL 绘制时需要自行实现顶点着色器和片元着色器:

  1. 顶点着色器的输入数据包括顶点属性(如位置、法线等)以及可选的采样器。输出数据是变换后的顶点位置(gl_Position)、正面朝向标志(gl_FrontFacing)以及其他临时变量
    • 首先,顶点着色器接收来自应用程序的顶点数据,这些数据通常包含顶点的位置、法线、纹理坐标等信息。顶点着色器的主要任务是对这些顶点进行变换,例如应用模型矩阵、视图矩阵和投影矩阵,以将顶点从对象空间转换到屏幕空间
    • 然后,顶点着色器会生成一个 gl_Position 变量,它是经过变换后的顶点位置,表示该顶点在屏幕上的位置。此外,顶点着色器还可以生成其他临时变量,用于后续的计算或传递给片元着色器
    • 接下来,顶点着色器会将变换后的顶点数据传递给片元着色器。在这个过程中,顶点着色器还会生成一些 Varying 变量,这些变量可以用来存储顶点之间的一些信息,比如纹理坐标或者光照参数
  2. 片元着色器输入数据包括从顶点着色器传递过来的 Varying 变量(如纹理坐标),以及一些内置变量(如点大小 gl_PointSize 和片段坐标 gl_FragCoord)。输出数据是一个颜色值(gl_FragColor),它将被用于最终的图像渲染
    • 片元着色器接收到顶点着色器传递来的 Varying 变量,以及一些内置变量,如 gl_FragCoord 和 gl_PointSize 等。片元着色器的主要任务是对每个像素的颜色进行计算,这可能涉及到纹理采样、光照计算、混合操作等
    • 最后,片元着色器会生成一个 gl_FragColor 变量,这是最终的颜色值,它会被用于绘制像素。如果需要,片元着色器也可以生成其他临时变量,用于后续的计算或传递给其他阶段

可编程管线

再来看 OpenGL 2.0 增加的可编程管线,它显示了图形渲染管线的基本工作流程:

2024-3-18.可编程管线

各部分职能:

  • 顶点着色器:是图形渲染管线的第一步,它主要负责处理顶点数据,包括顶点位置、法线、纹理坐标等。顶点着色器可以根据需要修改顶点数据,例如应用变换矩阵、添加顶点动画效果等
  • 图元装配:将顶点数据组装成图元的过程,常见的图元有三角形、矩形等。图元装配的结果是一组有序的顶点列表,这些顶点将被发送到光栅化阶段
  • 光栅化:将图元转换为像素的过程,它根据图元的几何形状和屏幕分辨率,确定哪些像素应该被填充。光栅化阶段还负责执行裁剪、深度测试、模板测试等操作
  • 片元着色器:是图形渲染管线中的一个重要步骤,它负责处理每个像素的颜色信息。片元着色器可以从顶点着色器获取数据,例如顶点位置、法线、纹理坐标等,然后根据这些数据计算出每个像素的颜色
  • 逐片元处理:在片元着色器之后的一个阶段,它负责执行各种后期处理效果,例如模糊、抗锯齿、景深等。这些效果可以在片元着色器之后应用,从而实现更复杂的视觉效果
  • 帧缓冲区:是图形渲染管线的最后一个阶段,它负责将渲染结果保存到屏幕上。帧缓冲区可以是硬件设备的一部分,也可以是软件模拟的内存区域

顶点着色器只是确定了图形的外部边界轮廓,而片元着色器才负责绘制内部的纹理(纹路)与颜色。

主要组件

最后来了解 Android OpenGL ES 包含的主要组件:

  • GLSurfaceView:继承自 SurfaceView,其内嵌的 Surface 专门负责 OpenGL 渲染。它支持:
    • 管理 Surface 与 EGL
    • 允许自定义渲染器 Render
    • 让渲染器在独立的线程里运作,和 UI 线程分离
    • 按需渲染(on-demand)和连续渲染(continuous)
  • EGL:OpenGL 是一个跨平台操作 GPU 的 API,但 OpenGL 需要与本地视窗系统进行交互,这就需要一个中间控制层,EGL 就是链接 OpenGL ES 和本地窗口系统的接口,引入 EGL 就是为了屏蔽不同平台上的区别(类似于上层通过 JNI 实现与本地代码的交互)

知道大致流程后可以进入代码阶段了。

3.2 初始化

首先在 AndroidManifest 中声明使用 OpenGL:

<uses-feature android:glEsVersion="0x00020000" android:required="true"/>

然后声明一个自定义 View 继承 GLSurfaceView,并设置 EGL 版本、渲染器和渲染模式:

// GLSurfaceView 继承了 SurfaceView,它自带一个线程 GLThread 进行渲染工作
class GLView(context: Context?, attrs: AttributeSet?) : GLSurfaceView(context, attrs) {

    constructor(context: Context?) : this(context, null)

    init {
        // 设置 EGL 版本
        setEGLContextClientVersion(2)
        // 设置渲染器
        setRenderer(GLRender())
        // 设置渲染模式:
        // RENDERMODE_WHEN_DIRTY 会在 Surface 被创建,或调用 requestRender() 时渲染
        // RENDERMODE_CONTINUOUSLY 会每隔一段时间自动渲染
        renderMode = RENDERMODE_WHEN_DIRTY
    }
}

再声明自定义渲染器 GLRender 继承 GLSurfaceView.Render:

class GLRender : GLSurfaceView.Renderer {
    // GLSurfaceView.Renderer start
    override fun onSurfaceCreated(gl: GL10?, config: EGLConfig?) {
        // 将画布清空为黑色
        GLES20.glClearColor(0f, 0f, 0f, 0f)
    }

    override fun onSurfaceChanged(gl: GL10?, width: Int, height: Int) {
    }

    override fun onDrawFrame(gl: GL10?) {
    }
    // GLSurfaceView.Renderer end
}

3.3 初始化三角形

我们新建一个 Triangle 来处理三角形的绘制,并且在 GLRender 绘画回调时,将回调同步给 Triangle:

class GLRender : GLSurfaceView.Renderer {
    
    override fun onSurfaceCreated(gl: GL10?, config: EGLConfig?) {
        // 将画布清空为黑色
        GLES20.glClearColor(0f, 0f, 0f, 0f)
        triangle = Triangle()
    }
    
    // 当外界调用 requestRender 时会触发本方法回调,
    // 类似于 invalidate 与 onDraw 的关系
    override fun onDrawFrame(gl: GL10?) {
        GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT or GLES20.GL_DEPTH_BUFFER_BIT)
        // 具体的渲染工作交给 Triangle 对象
        triangle.onDrawFrame(gl)
    }
}

接下来要确定被绘制的三角形是什么样的。比如我想画一个在 z = 0 这个平面的等腰直角三角形,如下图所示:

2024-3-17.OpenGL坐标系

那么该三角形的顶点数组声明如下:

	companion object {
        val triangleCoordinates = floatArrayOf(
            0.5f, 0.5f, 0.0f,
            -0.5f, -0.5f, 0.0f,
            0.5f, -0.5f, 0.0f
        )
    }

接下来需要创建顶点着色器和片元着色器了。首先我们在 AS 中安装 GLSL Support 插件,可以对 GLSL 进行高亮提示。然后在 /res/raw/ 目录下新建顶点着色器文件 vertex_shader.glsl:

// attribute 声明了包含四个浮点数的向量 vPosition
attribute vec4 vPosition;
// main 是着色器程序的入口点
void main() {
    // gl_Position 是语言内置变量,代表了顶点的位置
    gl_Position=vPosition;
}

然后是片元着色器 fragment_shader.glsl:

precision mediump float;
uniform vec4 vColor;
void main() {
    // 将颜色向量赋值给内置变量,OpenGL 就会根据该颜色绘制
    gl_FragColor=vColor;
}

接下来你可以将两个着色器的代码声明为字符串变量,或者通过 IO 流读取文件代码到字符串。这里为了方便我们采用第一种方式(后续 Demo 会采用第二种方式):

	private val vertexShaderCode = "attribute vec4 vPosition;\n" +
            "void main() {\n" +
            "    gl_Position=vPosition;\n" +
            "}"
    private val fragmentShaderCode = "precision mediump float;\n" +
            "uniform vec4 vColor;\n" +
            "void main() {\n" +
            "    gl_FragColor=vColor;\n" +
            "}"

最后在 Triangle 的构造方法中做如下的初始化工作:

	private var vertexBuffer: FloatBuffer
	private var program = 0
	init {
        // 在 GPU 中申请内存,三角形有 3 个顶点,每个顶点有 3 个维度,都是 4 个字节的 float 类型
        val byteBuffer = ByteBuffer.allocateDirect(triangleCoordinates.size * 4)
        // ByteBuffer 内元素排列顺序采用 Native 顺序
        byteBuffer.order(ByteOrder.nativeOrder())
        // GL 语言代码需要通过 vertexBuffer 传入 GPU
        vertexBuffer = byteBuffer.asFloatBuffer()
        vertexBuffer.put(triangleCoordinates)
        vertexBuffer.position(0)

        // 创建顶点着色器并在 GPU 中进行编译
        val vertexShader: Int = GLES20.glCreateShader(GLES20.GL_VERTEX_SHADER)
        GLES20.glShaderSource(vertexShader, vertexShaderCode)
        GLES20.glCompileShader(vertexShader)

        // 创建片元着色器并在 GPU 中进行编译
        val fragmentShader: Int = GLES20.glCreateShader(GLES20.GL_FRAGMENT_SHADER)
        GLES20.glShaderSource(fragmentShader, fragmentShaderCode)
        GLES20.glCompileShader(fragmentShader)

        // 创建 Program 并绑定两个着色器
        program = GLES20.glCreateProgram()
        GLES20.glAttachShader(program, vertexShader)
        GLES20.glAttachShader(program, fragmentShader)

        // 连接到着色器程序
        GLES20.glLinkProgram(program)
    }

3.4 绘制三角形

最后在 GLRender 传入的绘制回调方法 onDrawFrame() 中绘制三角形:

	private val colors = floatArrayOf(1.0f, 1.0f, 1.0f, 1.0f)

	fun onDrawFrame(gl: GL10?) {
        // 渲染
        GLES20.glUseProgram(program)
        // 获取 GPU 中 vPosition 变量的地址(实际上是一个 Native 指针)
        val positionHandle = GLES20.glGetAttribLocation(program, "vPosition")
        // 开启允许对变量读写,与关闭方法成对出现
        GLES20.glEnableVertexAttribArray(positionHandle)
        // 变量地址,数据尺寸,数据类型,是否正常,行跨度,偏移
        GLES20.glVertexAttribPointer(positionHandle, 3, GLES20.GL_FLOAT, false, 3 * 4, vertexBuffer)
        val colorHandle = GLES20.glGetUniformLocation(program, "vColor")
        GLES20.glUniform4fv(colorHandle, 1, colors, 0)
        // 让 OpenGL 绘制三角形
        GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, 3)
        // 关闭对变量读写
        GLES20.glDisableVertexAttribArray(positionHandle)
    }

实际上就是把实际的顶点坐标 vertexBuffer 赋值给 GPU 中的 vPosition 变量,将颜色数组写入到 GPU 的 vColor 变量。

运行程序确实画出了一个直角三角形,但是却不是我们想要的等腰直角三角形:

在这里插入图片描述

原因是 OpenGL 绘制的是三维图形,将三维图形绘制到 Android 屏幕这个二维平面上,是需要进行投影的。投影的目的是将三维物体的空间位置和形状转换为屏幕上的二维图像,使观察者可以从适当的视角观察和理解物体。在图形渲染过程中,投影通常是在三维物体的模型空间或世界空间与屏幕空间之间进行的转换。

投影方法主要有两种:

  1. 透视投影(Perspective Projection):透视投影模拟了人眼或相机在观察场景时的视角效果。它产生了近大远小的效果,使离观察者较远的物体显得较小。透视投影常用于创建逼真的三维效果,在虚拟现实、游戏和模拟等领域广泛应用。
  2. 正交投影(Orthographic Projection):正交投影是一种平行投影,保持了物体在不同距离上的大小不变。它独立于观察者的位置和角度,适用于需要保持物体大小和形状一致的场景,如工程制图、CAD 应用和二维游戏等。

投影过程涉及到三个要素:相机(观察物体的位置)、被观察的物体与投影面。

为了让最终看到的是等腰直角三角形,我们需要对顶点着色器中的顶点进行变换,变换方式就是对其左乘矩阵:

attribute vec4 vPosition;
uniform mat4 vMatrix;
void main() {
    gl_Position=vMatrix*vPosition;
}

现在问题转移到 vMatrix 是如何计算的。首先,我们让 GLRender 将 Surface 的变化同步给 Triangle:

	override fun onSurfaceChanged(gl: GL10?, width: Int, height: Int) {
        triangle.onSurfaceChanged(gl, width, height)
    }

在 Triangle 的 onSurfaceChanged() 中设置投影矩阵、相机矩阵,并计算出变换矩阵:

	private val viewMatrix = FloatArray(16)
    private val projectMatrix = FloatArray(16)
    private val mvpMatrix = FloatArray(16)

	fun onSurfaceChanged(gl: GL10?, width: Int, height: Int) {
        // 计算宽高比
        val ratio: Float = width.toFloat() / height
        // 设置投影矩阵
        Matrix.frustumM(projectMatrix, 0, -ratio, ratio, -1f, 1f, 3f, 120f)
        // 计算相机矩阵
        Matrix.setLookAtM(
            viewMatrix, 0,
            0f, 0f, 7f, // 摄像机的坐标
            0f, 0f, 0f, // 目标物的中心坐标
            0f, 1f, 0f // 相机看目标物的方向
        )
        // 计算变换矩阵,将 projectMatrix 与 viewMatrix 相乘
        Matrix.multiplyMM(mvpMatrix, 0, projectMatrix, 0, viewMatrix, 0)
    }

更新顶点着色器字符串并向 GPU 传入刚刚计算出的变换矩阵:

	private val vertexShaderCode = "attribute vec4 vPosition;\n" +
            "uniform mat4 vMatrix;\n" +
            "void main() {\n" +
            "    gl_Position=vMatrix*vPosition;\n" +
            "}"

	fun onDrawFrame(gl: GL10?) {
        // 渲染
        GLES20.glUseProgram(program)
        // 获取 GPU 中 vMatrix 变量的地址,并将 mvpMatrix 赋值给 vMatrix
        val matrixHandle = GLES20.glGetUniformLocation(program, "vMatrix")
        GLES20.glUniformMatrix4fv(matrixHandle, 1, false, mvpMatrix, 0)
        // 获取 GPU 中 vPosition 变量的地址(实际上是一个 Native 指针)
        val positionHandle = GLES20.glGetAttribLocation(program, "vPosition")
        // 开启允许对变量读写,与关闭方法成对出现
        GLES20.glEnableVertexAttribArray(positionHandle)
        // 变量地址,数据尺寸,数据类型,是否正常?,行跨度,偏移
        GLES20.glVertexAttribPointer(positionHandle, 3, GLES20.GL_FLOAT, false, 3 * 4, vertexBuffer)
        val colorHandle = GLES20.glGetUniformLocation(program, "vColor")
        GLES20.glUniform4fv(colorHandle, 1, colors, 0)
        GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, 3)
        // 关闭对变量读写
        GLES20.glDisableVertexAttribArray(positionHandle)
    }

这样就能画出一个等腰直角三角形:

在这里插入图片描述

最后总结一下绘制步骤:

  • 创建顶点数组
  • 创建顶点着色器和片元着色器
  • 将上层定义的顶点数组和颜色数组通过 OpenGL ES 提供的 API 接口传递给 GL 语言的变量

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1638304.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一个C++小程序调试过程记录

Top 20 C Projects With Source Code [2024 Update]https://www.interviewbit.com/blog/cpp-projects/ 这个网页有一些简单的C程序的源码&#xff0c;闲来无事&#xff0c;把第一个程序&#xff08;Bookshop Management System Using C&#xff09;的源码下载了下来。 源文件…

在 Vue.js 2.x 中使用 WebSocket 与 Spring Boot 2.7 实现通信

前言&#xff1a;近期在研究远程遥控空调&#xff0c;需要通过接口记录前端的一些动作&#xff0c;那么前端需要实现主动通知接口&#xff0c;并接收来自接口的通知。 一、方案 通过在Vue.js 2.x 中使用 WebSocket 与 Spring Boot 2.7 实现通信&#xff0c;使用 STOMP&#xf…

【openLooKeng集成Hive连接器完整过程】

【openLooKeng集成Hive连接器完整过程】 一、摘要二、正文2.1 环境说明2.2 Hadoop安装2.2.1. 准备工作2.2.2 在协调节点coordinator上进行安装hadoop2.2.3、将Hadoop安装目录分发到从节点worker2.2.4、在协调节点coordinator上启动hadoop集群2.3 MySQL安装2.4 Hive安装及基本操…

LabVIEW智能变电站监控系统设计与实现

LabVIEW智能变电站监控系统设计与实现 随着电力系统和智能化技术的快速发展&#xff0c;建立一个高效、可靠的变电站监控系统显得尤为重要。通过分析变电站监控系统的需求&#xff0c;设计了一个基于LabVIEW软件的监控平台。该平台利用虚拟仪器技术、传感器技术和无线传输技术…

从C到Py:Python的异常处理

本篇文章中我们将简单讲解一下有关Python中的异常处理&#xff0c;既有在代码中有显式表达的异常处理语法&#xff0c;还包括Pycharm手动调试的简单操作。 目录 Python异常处理 1、try-except结构 ​编辑 2、try-except-else结构 raise抛出异常 常见异常类型 Pycharm程序…

神经网络中常见的激活函数:理解与实践

神经网络中常见的激活函数&#xff1a;理解与实践 在神经网络中&#xff0c;激活函数是一个非常重要的组成部分&#xff0c;它为神经元引入了非线性特性&#xff0c;使得神经网络可以拟合各种复杂的函数关系。本文将介绍9种常见的激活函数&#xff0c;包括它们的概述、公式以及…

安卓中级控件(图形、选择按钮、文本输入、对话框)

图形定制 图形Drawable Android把所有能够显示的图形都抽象为Drawable类&#xff08;可绘制的&#xff09;。这里的图形不止是图片&#xff0c;还包括色块、画板、背景等。 包含图片在内的图形文件放在res目录的各个drawable目录下&#xff0c;其中drawable目录一般保存描述性…

相机知识的补充

一&#xff1a;镜头 1.1MP的概念 相机中MP的意思是指百万像素。MP是mega pixel的缩写。mega意为一百万&#xff0c;mega pixel 指意为100万像素。“像素”是相机感光器件上的感光最小单位。就像是光学相机的感光胶片的银粒一样&#xff0c;记忆在数码相机的“胶片”&#xff…

windows 驱动开发-DMA技术(二)

前面描述了DMA技术中适配器相关的部分以及DMA的分类&#xff0c;接下来看一下系统具体在支持两种DMA时候的操作的细微差别。 此处解释一下Scatter/Gather&#xff0c;这个也翻译为散点/收集&#xff0c;是指指示设备能够读取或写入内存中的任何区域&#xff0c;而不仅仅是特定…

YOLO系列改进,自研模块助力涨点

目录 一、原理 二、代码 三、添加到YOLOv5中 一、原理 论文地址:

LiveCD镜像文件的定制化

最近想要定制化一款属于自己的LiveCD镜像文件&#xff0c;并且里边封装好所需要的软件程序&#xff0c;本文将会记录具体的操作步骤&#xff0c;供有需要的人参考借鉴。 环境说明&#xff1a; 环境配置说明配置参数编码环境Centos7.9LiveCD文件CentOS-7-livecd-x86_64.iso 附…

袁庭新ES系列18节|Spring Data Elasticsearch高级

前言 这一章节袁老师将带领同学们来学习Spring Data Elasticsearch高级操作相关的内容。我们继续来探索SDE是如何将原始操作Elasticsearch的客户端API进行封装的&#xff0c;以及通过Spring Data Elasticsearch如何来操作ES。准备好了吗&#xff1f;我们继续来探索ES的内容。 …

探索AI工具的巅峰:个人体验与深度剖析

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

【Go语言快速上手(六)】管道, 网络编程,反射,用法讲解

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Go语言专栏⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学习更多Go语言知识   &#x1f51d;&#x1f51d; GO快速上手 1. 前言2. 初识管道3. 管…

自定义SpringBoot的starter

案例需求&#xff1a;自定义redis-stater。要求当导入redis坐标时&#xff0c;SpringBoot自动创建Jedis的Bean。 实现步骤&#xff1a; 1、创建redis-spring-boot-autoconfigure模块 2、创建redis-spring-boot-starter模块&#xff0c;依赖redis-spring-boot-autoconfigure的…

Codeforces Round 938 (Div. 3)H-The Most Reckless Defense

来源 题目 You are playing a very popular Tower Defense game called "Runnerfield 2". In this game, the player sets up defensive towers that attack enemies moving from a certain starting point to the players base. You are given a grid of size nm&a…

Delta lake with Java--利用spark sql操作数据2

上一篇文章尝试了建库&#xff0c;建表&#xff0c;插入数据&#xff0c;还差删除和更新&#xff0c;所以在这篇文章补充一下&#xff0c;代码很简单&#xff0c;具体如下&#xff1a; import org.apache.spark.sql.SaveMode; import org.apache.spark.sql.SparkSession;publi…

DRF中的请求入口分析及request对象分析

DRF中的请求入口分析及request对象分析 django restframework框架是在django的基础上又给我们提供了很多方便的功能&#xff0c;让我们可以更便捷基于django开发restful API 1 drf项目 pip install django pip install djangorestframework1.1 核心配置 INSTALLED_APPS [d…

【源码阅读】Golang中的go-sql-driver库源码探究

文章目录 前言一、go-sql-driver/mysql1、驱动注册&#xff1a;sql.Register2、驱动实现&#xff1a;MysqlDriver3、RegisterDialContext 二、总结 前言 在上篇文章中我们知道&#xff0c;database/sql只是提供了驱动相关的接口&#xff0c;并没有相关的具体实现&#xff0c;具…

PG数据库结构与oracle比较

1.数据库集簇逻辑结构 数据库集簇概念&#xff1a;一个大的数据库是由若干个小的数据库组成&#xff0c;实现数据的隔离存放&#xff0c;在概念上应该是与mysql一样的 在mysql中可以用show database列出数据库 PG中用\l 数据库对象存放在数据库中&#xff1a; PG中的所有数据…