Android OPENGL 第二天:通过NDK实现GL的绘制(

2020-08-17  本文已影响0人  阿里高级软件架构师

    从今天开始我学习的GL知识都通过使用C++进行开发,需要用到安卓的NDK

    今天先封装一个我们自己的工具类

    C++的语法还是比较难理解,可能是我经常用Java习惯了,突然要我去做C++,还是有一点难为人,ok,开始学习

    首先GL的绘制和使用需要用到下面的几个重要工具:

    EGLDisplay:这个是GL现实的显存,用来连接显示器的

    EGLSurface:这个是buffer的缓冲区

    EGLConfig:这个就从字面上可以理解为GL的配置信息

    EGLContext:因为需要用到与显存的buffer交换,需要共享上下文,这个还是很重要的。

    我们首先做一个工具类,就叫EGLHelper好了:

class WlEglHelper {

public:

    EGLDisplay  mEglDisplay;

    EGLSurface  mEglSurface;

    EGLConfig  mEglConfig;

    EGLContext mEglContext;

}

那定义好了基础变量,还得定义一个方法对GL进行初始化,那就命名 initEgl()好了,但是你还得存入一个EGLNativeWindowType 的参数,因为这个是为了让gl展示的window的类型,因为现实的类型不只是这种,可能还有其他的类型,不过现在就用到了这种类型

    那现在就写这个方法的逻辑:

思路:首先你得获取他的展示容器,GL的容器的ID的名称是:

EGL_DEFAULT_DISPLAY

那然后就是判断你有没有获取容器成功咯!所以就通过判断

if(mEglDisplay ==EGL_NO_DISPLAY){

LOGE("eglGetDisplay error");

    return -1;

}

然后接下来就是对GL进行初始化,并且进行判断是否成功:

EGLint *version =new EGLint[2];

if(!eglInitialize(mEglDisplay, &version[0], &version[1])){

LOGE("eglInitialize error");

    return -1;

}

然后接下来就声明gl的一些基本属性值:

const EGLint attribs[] = {

EGL_RED_SIZE, 8,

        EGL_GREEN_SIZE, 8,

        EGL_BLUE_SIZE, 8,

        EGL_ALPHA_SIZE, 8,

        EGL_DEPTH_SIZE, 8,

        EGL_STENCIL_SIZE, 8,

        EGL_RENDERABLE_TYPE,

        EGL_OPENGL_ES2_BIT,

        EGL_NONE

};

这几个变量的意义我说明一下:

EGL_RED_SIZE, 8

的意思是RED这个在颜色缓冲区红色分量的位数为8,

那后面的EGL_GREEN_SIZE,EGL_BLUE_SIZE就是绿色和蓝色的在颜色缓冲区的位数,后面的透明值(EGL_ALPHA_SIZE)和深度缓冲区(EGL_DEPTH_SIZE)的位数,模板缓冲区(EGL_STENCIL_SIZE)的位数,支持渲染的接口(EGL_RENDERABLE_TYPE),那最后的那个是做什么的呢?最后那个是用来做结束语的,就是标志着这个数组结束了。

然后就是通过eglChooseConfig来进行对GL的配置了

EGLint num_config;

if(!eglChooseConfig(mEglDisplay, attribs, NULL, 1, &num_config)){

LOGE("eglChooseConfig  error 1");

    return -1;

}

然后就是要获取上下文,这个的配置项的内容是:

int attrib_list[] = {

EGL_CONTEXT_CLIENT_VERSION, 2,

        EGL_NONE

};

这个配置就比较简单了,第一个参数是GL上下文的客户端版本号,然后后面那个就不用说了,

然后实现的方法是:

mEglContext = eglCreateContext(mEglDisplay, mEglConfig, EGL_NO_CONTEXT, attrib_list);

if(mEglContext ==EGL_NO_CONTEXT){

LOGE("eglCreateContext  error");

    return -1;

}

然后接下来就是创建显示的界面了:

mEglSurface = eglCreateWindowSurface(mEglDisplay, mEglConfig, window, NULL);

if(mEglSurface ==EGL_NO_SURFACE){

LOGE("eglCreateWindowSurface  error");

    return -1;

}

然后你做完了GL的对应的配置,你要如何启动他们呢?我们目前有上下文,有显示界面,有配置项,有显示模板,但是想要他们组合在一起,并且实现上下问的切换,需要用到一个很重要的方法:

if(!eglMakeCurrent(mEglDisplay, mEglSurface, mEglSurface, mEglContext)){

LOGE("eglMakeCurrent  error");

    return -1;

}

就是makeCurrent(),这个方法是用来对上下文进行切换,OPENGL才可以起作用的,这个还是蛮重要的。

那做完了这些,那是不是还得让他们不停的切换,这个切换是与显存进行buffer的交换,那还是得使用eglSwapBuffers()进行display和surface进行buffer的交换:

if(mEglDisplay !=EGL_NO_DISPLAY &&mEglSurface !=EGL_NO_SURFACE){

if(eglSwapBuffers(mEglDisplay, mEglSurface))

return 0;

  }

}

return -1;

好了,我们来总结一下实现思路:GL其实就是显卡暴露出来的一系列的接口,是可以通过他直接操作显卡进行显示你要的图像,然后显卡一般是不处理视频的,只处理图片,然后这个在开发的时候我们称做纹理,然后还有一个很重要的知识点,就是所有的纹理图案都是通过三角形进行破解的,然后你就会问了,为什么不是四边形呢?或者其他的图形呢?因为三角形是可以拼接成为很多图形,比如四边形也是两个三角形拼接的,五边形,六边形也是,然后就是你会发现,在gl上绘制一条直线,其实他不是很直,他的线条上面会有很多锯齿,那是因为绘制一条直线,需要经过很多像素点,但是这个像素点又不在同一条直线上,都是在一条直线的两侧或者部分在中间,所有你要经历大部分的点,因此就有锯齿。

    然后你要显示图像的话,需要在内存上开辟一块区域用来进行计算,手机没有显存的这个说法,因为所有的计算都是在内存的某个区域进行计算的,而且他是最善于计算的,不会做其他的处理,包括存储什么的!所有他处理图像的速度还是很快的,然后他在显卡中会产生buffer,然后gl就是通过buffer和显卡进行交互来显示图像的,这个buffer你也可以拿来转码转换为视频存储在本地,这个我们后面再说

    上面的流程还是蛮基础的,希望大家看完可以收获一点东西!

上一篇下一篇

猜你喜欢

热点阅读