Android OpenGL ES2.0基础(二、画个三角形)

2024-04-07 07:32

本文主要是介绍Android OpenGL ES2.0基础(二、画个三角形),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、概要

上节中了解了OpenGLES最简单的使用OpenGLES2.0基础,这节在深入一个层次了解怎么去画一个三角形。对于OpenGL绘图,最主要的就是Renderer的处理。因为GLSurfaceView相当于一个View,只是显示图像,而Renderer就是GLSurfaceView要显示的图像。在Renderer中主要有三个函数onSurfaceCreated(),onSurfaceCreated()和onDrawFrame()。其中前两个函数主要用于初始化逻辑,绘图主要是在onDrawFrame()方法里面。

二、相关要点

1.绘制方式

OpenGLES图元的绘制方式有GL_POINTS 、GL_LINES、 GL_LINE_LOOP 、GL_LINE_STRIP、GL_TRIANGLES 、GL_TRIANGLE_STRIP 、GL_TRIANGLE_FAN 这几种,每个都有自己独特的作用。在使用时根据自己的需求选择不同的绘制方式进行绘制。

绘制方式解释
GL_POINTS
GL_LINES线段
GL_LINE_STRIP多段线
GL_LINE_LOOP线圈
GL_TRIANGLES三角形
GL_TRIANGLE_STRIP三角形条带
GL_TRIANGLE_FAN三角形扇

1. 非索引法: GLES20.glDrawArrays(GLES20.GL_POINTS, 0, vCount);
2. 索引法 :GLES20.glDrawElements(GLES20.GL_POINTS, iCount, GLES20.GL_UNSIGNED_BYTE, mIndexBuffer);

(1).GL_LINES(绘制直线)

以(v0,v1),(v2,v3),(v4,v5)的形式绘制直线。(当绘制的点只有两个时,GL_LINES与GL_LINE_STRIP绘线方式没有差异)

GLES20.glDrawArrays(GLES20.GL_LINES, 0, count);

(2).GL_LINE_STRIP(绘制直线)

以(v0,v1),(v1,v2),(v2,v3),(v3,v4),(v4,v5)的形式绘制直线。GL_LINE_LOOP以(v0,v1),(v1,v2),(v2,v3),(v3,v4),(v4,v5),(v5,v0)的形式绘制直线,比GL_LINE_STRIP多一条(v5,v0)的直线。

GLES20.glDrawArrays(GLES20.GL_LINE_STRIP, 0, count);

(3).GL_POINTS(绘制点)

GLES20.glDrawArrays(GLES20.GL_POINTS, 0, count);

(4).GL_TRIANGLES(绘制三角形)

每三个顶之间绘制三角形,之间不连接。

GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, count);

(5).GL_TRIANGLE_STRIP(绘制三角形)

它是由最少3个点构成(正好3个点就是一个GL_TRIANGLES)每增加1个点,新增的点会和之前已有的两个点构成新的三角形,依次类推。

GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, count);

(6).GL_TRIANGLE_FAN(绘制三角形)

以(v0,v1,v2),(v0,v2,v3),(v0,v3,v4)的形式绘制三角形。

GLES20.glDrawArrays(GLES20.GL_TRIANGLE_FAN, 0, count);

2.OpenGL矩阵变换

通过模拟人眼观察物体, 可能引起屏幕图像改变的几种行为和分别对应的几种OpenGL变换,每种变换都可以由相应的Matrix静态方法生成相应的变换矩阵。 坐标系原点在屏幕中央,z轴正方向垂直屏幕向外。

行为变换记录变换方式的放法记录变换方式的[4*4]矩阵
眼睛相对物体的位置改变视图变换Matrix.setLookAtM(mVMatrix,0(偏移量),cx(相机位置x), cy, cz,tx(观察点位置x), ty, tz,upx(顶部朝向x), upy, upz)mVMatrix[16]
物体平移模型变换Matrix.translateM(mMMatrix,0(偏移量),x(平移量x), y, z)mMMatrix[16]
物体按坐标缩放比例缩放模型变换Matrix.scaleM(mMMatrix,sx(缩放因子),sy, sz)mMMatrix[16]
物体旋转模型变换Matrix.rotateM(mMMatrix,0(偏移量),angle(旋转角度),x(需要旋转的轴), y, z)mMMatrix[16]
凸透镜眼睛投影变换Matrix.frustumM(mPMatrix,0(偏移量),left,right,bottom,top,near,far//near>0)mPMatrix[16]
平面透镜眼睛投影变换Matrix.orthoM(mPMatrix,0(偏移量),left,right,bottom,top,near,far//near>0)mPMatrix[16]

1. 变换技巧: 利用栈存储变换矩阵的算法可实现复杂场景的图形变换和提高渲染效率。
2. 变换顺序: 矩阵相乘是讲究顺序的, 比如先平移再缩放 和 先缩放再平移的效果是不同的。

三、绘制流程

  1. 定义物体顶点信息
  2. 编辑着色器代码
  3. 编译连接着色器程序
  4. 物体顶点信息传入着色器
  5. GLSurfaceView加载Renderer
  6. Activity加载GLSurfaceView
  7. 渲染物体输出屏幕

1.定义物体顶点信息

物体的顶点信息一般包括:(1)三维坐标信息; (2)三维法向量信息;(3)颜色信息或二维贴图坐标信息;复杂物体的坐标通常通过特定的软件绘制生成顶点信息文件,然后进行解读;

2.编辑着色器代码

着色器语言编写的代码, 通常后缀名.sh, 语法和C相似95%以上, 是OpenGLES2.0的可编程渲染通道, 增加了编程难度, 但是能实现更绚丽多彩的效果。一般编写好的着色器代码放在android工程目录下的assets文件, assets常用轻量资源的读写。

3.编译连接着色器

从assets获取着色器脚本, 并返回脚本字符串的过程

// 加载顶点着色器的脚本内容
mVertexShader = ShaderUtil.loadFromAssetsFile("graph_color_and_texture_vertex.shl",mContext.getResources());
// 加载片元着色器的脚本内容
mFragmentShader = ShaderUtil.loadFromAssetsFile("graph_color_and_texture_frag.shl",mContext.getResources());

加载片元着色器和顶点着色器

// 加载顶点着色器
int vertexShader = loadShader(GLES20.GL_VERTEX_SHADER, vertexSource);
if (vertexShader == 0) {return 0;
}
// 加载片元着色器
int pixelShader = loadShader(GLES20.GL_FRAGMENT_SHADER, fragmentSource);
if (pixelShader == 0) {return 0;
}
// 创建程序
int program = GLES20.glCreateProgram();
// 若程序创建成功则向程序中加入顶点着色器与片元着色器
if (program != 0) {// 向程序中加入顶点着色器GLES20.glAttachShader(program, vertexShader);checkGlError("glAttachShader");// 向程序中加入片元着色器GLES20.glAttachShader(program, pixelShader);checkGlError("glAttachShader");// 链接程序GLES20.glLinkProgram(program);// 存放链接成功program数量的数组int[] linkStatus = new int[1];// 获取program的链接情况GLES20.glGetProgramiv(program, GLES20.GL_LINK_STATUS, linkStatus, 0);// 若链接失败则报错并删除程序if (linkStatus[0] != GLES20.GL_TRUE) {Log.e("ES20_ERROR", "Could not link program: ");Log.e("ES20_ERROR", GLES20.glGetProgramInfoLog(program));GLES20.glDeleteProgram(program);program = 0;}
}

4. 物体顶点信息传入着色器

生成顶点信息, 用FloatBuffer类存储

ByteBuffer bbv = ByteBuffer.allocateDirect(triangleCoords.length * 4);//一个float 4个字节
bbv.order(ByteOrder.nativeOrder());
mVertexBuffer = bbv.asFloatBuffer();
mVertexBuffer.put(triangleCoords);//存入信息
mVertexBuffer.position(0);//设置起始位置为0;

获取着色器参数ID

maPositionHandle = GLES20.glGetAttribLocation(mProgram, "vPosition");

传入数据

GLES20.glVertexAttribPointer(maPositionHandle, 3, GLES20.GL_FLOAT, false, 3 * 4, mVertexBuffer);

使用数据

GLES20.glEnableVertexAttribArray(maPositionHandle);

5. GLSurfaceView加载Renderer

初始化GLSurfaceView

//设置版本
setEGLContextClientVersion(2);
//创建渲染器实例
mRenderer = new MyRenderer();
// 设置渲染器
setRenderer(mRenderer);
//设置渲染模式
setRenderMode(GLSurfaceView.RENDERMODE_WHEN_DIRTY);

初始化Renderer

    /*** 仅调用一次,用于设置view的OpenGL ES环境*/@Overridepublic void onSurfaceCreated(GL10 gl, EGLConfig config) {//设置背景色GLES20.glClearColor(0.0f, 0.0f, 0.0f, 1.0f);triangle = new Triangle();}/*** 当view的几何形状发生变化时调用,比如设备从竖屏变为横屏*/@Overridepublic void onSurfaceChanged(GL10 gl, int width, int height) {GLES20.glViewport(0, 0, width, height);float ratio = (float) width / height;Matrix.frustumM(mProjectionMatrix, 0, -ratio, ratio, -1, 1, 3, 7);}/*** 每次重绘view时调用*/@Overridepublic void onDrawFrame(GL10 gl) {float[] scratch = new float[16];// 绘制背景GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT | GLES20.GL_DEPTH_BUFFER_BIT);Matrix.setLookAtM(mViewMatrix, 0, 0, 0, -3, 0f, 0f, 0f, 0f, 1.0f, 0.0f);Matrix.multiplyMM(mMVPMatrix, 0, mProjectionMatrix, 0, mViewMatrix, 0);Matrix.setRotateM(mRotationMatrix, 0, mAngle, 0, 0, 1.0f);Matrix.multiplyMM(scratch, 0, mMVPMatrix, 0, mRotationMatrix, 0);// 重绘背景色triangle.draw(scratch);}

7. 渲染物体输出屏幕

//编译OpenGL ES shader和链接program
ProgramFactory.initFactory(getApplicationContext());mySurfaceView = new MySurfaceView(this);
setContentView(mySurfaceView);

四、运行效果

这里写图片描述

五、全部代码

1.在Manifests中添加

<uses-feature android:glEsVersion="0x00020000" android:required="true" />
<supports-gl-texture android:name="GL_OES_compressed_ETC1_RGB8_texture" />
<supports-gl-texture android:name="GL_OES_compressed_paletted_texture" />

2.MyGLSurfaceView继承自SurfaceView

import android.content.Context;
import android.opengl.GLSurfaceView;
import android.view.MotionEvent;/*** Created by Administrator on 2017/3/8.* description :*/public class MyGLSurfaceView extends GLSurfaceView {MyRenderer mRenderer;public MyGLSurfaceView(Context context) {super(context);init();}private void init(){// 创建一个OpenGL ES 2.0 contextsetEGLContextClientVersion(2);//创建渲染器实例mRenderer = new MyRenderer();// 设置渲染器setRenderer(mRenderer);//设置渲染模式setRenderMode(GLSurfaceView.RENDERMODE_WHEN_DIRTY);}private final float TOUCH_SCALE_FACTOR = 180.0f / 320;private float mPreviousX;private float mPreviousY;@Overridepublic boolean onTouchEvent(MotionEvent e) {float x = e.getX();float y = e.getY();switch (e.getAction()) {case MotionEvent.ACTION_MOVE:float dx = x - mPreviousX;float dy = y - mPreviousY;if (y > getHeight() / 2) {dx = dx * -1 ;}if (x < getWidth() / 2) {dy = dy * -1 ;}mRenderer.setAngle(mRenderer.getAngle() +((dx + dy) * TOUCH_SCALE_FACTOR));requestRender();}mPreviousX = x;mPreviousY = y;return true;}
}

3.MyRenderer实现了GLSurfaceView.Renderer接口

import android.opengl.GLES20;
import android.opengl.GLSurfaceView;
import android.opengl.Matrix;import javax.microedition.khronos.egl.EGLConfig;
import javax.microedition.khronos.opengles.GL10;/*** Created by Administrator on 2017/3/8.* description :*/
public class MyRenderer implements GLSurfaceView.Renderer {Triangle triangle;private final float[] mMVPMatrix = new float[16];private final float[] mProjectionMatrix = new float[16];private final float[] mViewMatrix = new float[16];private final float[] mRotationMatrix = new float[16];private float mAngle;/*** 仅调用一次,用于设置view的OpenGL ES环境*/@Overridepublic void onSurfaceCreated(GL10 gl, EGLConfig config) {//设置背景色GLES20.glClearColor(0.0f, 0.0f, 0.0f, 1.0f);triangle = new Triangle();}/*** 当view的几何形状发生变化时调用,比如设备从竖屏变为横屏*/@Overridepublic void onSurfaceChanged(GL10 gl, int width, int height) {GLES20.glViewport(0, 0, width, height);float ratio = (float) width / height;Matrix.frustumM(mProjectionMatrix, 0, -ratio, ratio, -1, 1, 3, 7);}/*** 每次重绘view时调用*/@Overridepublic void onDrawFrame(GL10 gl) {float[] scratch = new float[16];// 绘制背景GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT | GLES20.GL_DEPTH_BUFFER_BIT);Matrix.setLookAtM(mViewMatrix, 0, 0, 0, -3, 0f, 0f, 0f, 0f, 1.0f, 0.0f);Matrix.multiplyMM(mMVPMatrix, 0, mProjectionMatrix, 0, mViewMatrix, 0);Matrix.setRotateM(mRotationMatrix, 0, mAngle, 0, 0, 1.0f);Matrix.multiplyMM(scratch, 0, mMVPMatrix, 0, mRotationMatrix, 0);// 重绘背景色triangle.draw(scratch);}public void setAngle(float angle) {this.mAngle = angle;}public float getAngle() {return mAngle;}
}

3.Triangle主要处理画三角形逻辑

import android.opengl.GLES20;import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.FloatBuffer;/*** Created by Administrator on 2017/3/8.* description :*/
public class Triangle {// GL 绘图private FloatBuffer mVertexBuffer; // 顶点坐标数据缓冲private int mProgram; // 自定义渲染管线程序idprivate int mMVPMatrixHandle;// 总变换矩阵引用idprivate int maPositionHandle; // 顶点位置属性引用idprivate int maColorHandle; // 顶点颜色属性引用id// 设置每个顶点的坐标数static final int COORDS_PER_VERTEX = 3;// 设置三角形顶点数组static float triangleCoords[] = { // 默认按逆时针方向顺序绘制0.0f, 0.622008459f, 0.0f,   // 顶-0.5f, -0.311004243f, 0.0f,   // 左底0.5f, -0.311004243f, 0.0f    // 右底};private final int vertexCount = triangleCoords.length / COORDS_PER_VERTEX;private final int vertexStride = COORDS_PER_VERTEX * 4; // 4 bytes per vertex// 设置图形的RGB值和透明度static float color[] = {0.63671875f, 0.76953125f, 0.22265625f, 1.0f};public Triangle() {initVertexData();initShader();}// 初始化顶点坐标与着色数据的方法public void initVertexData() {// 如果数据只有一个的时候会出错,mVertexBuffer和mColorBuffer容错10ByteBuffer bbv = ByteBuffer.allocateDirect(triangleCoords.length * 4);//一个float 4个字节bbv.order(ByteOrder.nativeOrder());///前期设置mVertexBuffer = bbv.asFloatBuffer();//作为中介开辟FloatBuffermVertexBuffer.put(triangleCoords);mVertexBuffer.position(0);}// 初始化着色器public void initShader() {// 加载顶点着色器的脚本内容mProgram = ProgramFactory.shared().getProgram();// 获取程序中顶点位置属性引用idmaPositionHandle = GLES20.glGetAttribLocation(mProgram, "vPosition");// 获取程序中顶点颜色属性引用idmaColorHandle = GLES20.glGetUniformLocation(mProgram, "vColor");// 获取程序中总变换矩阵引用idmMVPMatrixHandle = GLES20.glGetUniformLocation(mProgram, "uMVPMatrix");}public void draw(float[] mvpMatrix) {// 制定使用某套shader程序GLES20.glUseProgram(mProgram);// 允许顶点位置数据数组GLES20.glEnableVertexAttribArray(maPositionHandle);// 为画笔指定顶点位置数据GLES20.glVertexAttribPointer(maPositionHandle, 3, GLES20.GL_FLOAT, false, vertexStride, mVertexBuffer);//GLES20.glUniform4fv(maColorHandle, 1, color, 0);//mMVPMatrixHandle = GLES20.glGetUniformLocation(mProgram, "uMVPMatrix");//GLES20.glUniformMatrix4fv(mMVPMatrixHandle, 1, false, mvpMatrix, 0);//GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, vertexCount);//GLES20.glDisableVertexAttribArray(maPositionHandle);}}

4.在MainActivity中使用

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;/*** Created by Administrator on 2017/3/8.* description :*/
public class MainActivity extends AppCompatActivity {MyGLSurfaceView mySurfaceView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);//编译OpenGL ES shader和链接programProgramFactory.initFactory(getApplicationContext());mySurfaceView = new MyGLSurfaceView(this);setContentView(mySurfaceView);}@Overrideprotected void onPause() {super.onPause();mySurfaceView.onPause();}@Overrideprotected void onResume() {super.onResume();mySurfaceView.onResume();}
}

5.以及在assets目录下的两个着色器脚本

graph_color_and_texture_frag.shl

precision mediump float;
uniform vec4 vColor;
void main() {gl_FragColor = vColor;
}

graph_color_and_texture_vertex.shl

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

6.下面贴上两个OpenGL工具类

ProgramFactory

import android.content.Context;/*** Created by Administrator on 2017/3/8.* description :*/
public class ProgramFactory {private static Context mContext = null;String mVertexShader;String mFragmentShader;int mMixProgram;public static void initFactory(Context context){mContext = context;}private ProgramFactory() {// 加载顶点着色器的脚本内容mVertexShader = ShaderUtil.loadFromAssetsFile("graph_color_and_texture_vertex.shl",mContext.getResources());// 加载片元着色器的脚本内容mFragmentShader = ShaderUtil.loadFromAssetsFile("graph_color_and_texture_frag.shl",mContext.getResources());// 基于顶点着色器与片元着色器创建程序mMixProgram = ShaderUtil.createProgram(mVertexShader, mFragmentShader);}private static ProgramFactory mInstance;public static ProgramFactory shared() {if(mContext == null){throw new IllegalStateException("ProgramFactory.initFactory() method is not transfer : mContext = null");}if (mInstance == null) {mInstance = new ProgramFactory();}return mInstance;}public static void recycle() {mInstance = null;}public int getProgram() {return mMixProgram;}
}

ShaderUtil

import android.content.res.Resources;
import android.opengl.GLES20;
import android.util.Log;import java.io.ByteArrayOutputStream;
import java.io.InputStream;
/*** Created by Administrator on 2017/3/8.* description :加载顶点Shader与片元Shader的工具类*/
public class ShaderUtil {/*** 加载制定shader的方法** @param shaderType*            shader的类型 GLES20.GL_VERTEX_SHADER GLES20.GL_FRAGMENT_SHADER* @param source*            shader的脚本字符串* @return*/public static int loadShader(int shaderType, String source) {// 创建一个新shaderint shader = GLES20.glCreateShader(shaderType);// 若创建成功则加载shaderif (shader != 0) {if (source != null) {// 加载shader的源代码GLES20.glShaderSource(shader, source);}// 编译shaderGLES20.glCompileShader(shader);// 存放编译成功shader数量的数组int[] compiled = new int[1];// 获取Shader的编译情况GLES20.glGetShaderiv(shader, GLES20.GL_COMPILE_STATUS, compiled, 0);// 若编译失败则显示错误日志并删除此shaderif (compiled[0] == 0) {// Log.e("ES20_ERROR", "Could not compile shader " + shaderType + ":");// Log.e("ES20_ERROR", GLES20.glGetShaderInfoLog(shader));GLES20.glDeleteShader(shader);shader = 0;}}return shader;}// 创建shader程序的方法public static int createProgram(String vertexSource, String fragmentSource) {// 加载顶点着色器int vertexShader = loadShader(GLES20.GL_VERTEX_SHADER, vertexSource);if (vertexShader == 0) {return 0;}// 加载片元着色器int pixelShader = loadShader(GLES20.GL_FRAGMENT_SHADER, fragmentSource);if (pixelShader == 0) {return 0;}// 创建程序int program = GLES20.glCreateProgram();// 若程序创建成功则向程序中加入顶点着色器与片元着色器if (program != 0) {// 向程序中加入顶点着色器GLES20.glAttachShader(program, vertexShader);checkGlError("glAttachShader");// 向程序中加入片元着色器GLES20.glAttachShader(program, pixelShader);checkGlError("glAttachShader");// 链接程序GLES20.glLinkProgram(program);// 存放链接成功program数量的数组int[] linkStatus = new int[1];// 获取program的链接情况GLES20.glGetProgramiv(program, GLES20.GL_LINK_STATUS, linkStatus, 0);// 若链接失败则报错并删除程序if (linkStatus[0] != GLES20.GL_TRUE) {Log.e("ES20_ERROR", "Could not link program: ");Log.e("ES20_ERROR", GLES20.glGetProgramInfoLog(program));GLES20.glDeleteProgram(program);program = 0;}}return program;}// 检查每一步操作是否有错误的方法public static void checkGlError(String op) {int error;while ((error = GLES20.glGetError()) != GLES20.GL_NO_ERROR) {// Log.e("ES20_ERROR", op + ": glError " + error);throw new RuntimeException(op + ": glError " + error);}}// 从sh脚本中加载shader内容的方法public static String loadFromAssetsFile(String fname, Resources r) {String result = null;try {InputStream in = r.getAssets().open(fname);int ch = 0;ByteArrayOutputStream baos = new ByteArrayOutputStream();while ((ch = in.read()) != -1) {baos.write(ch);}byte[] buff = baos.toByteArray();baos.close();in.close();result = new String(buff, "UTF-8");result = result.replaceAll("\\r\\n", "\n");} catch (Exception e) {e.printStackTrace();}return result;}
}

以上即为使用OpenGLES20画三角形的全部代码

相关链接:

源码下载

Shader相关介绍

matrix详解

这篇关于Android OpenGL ES2.0基础(二、画个三角形)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/881994

相关文章

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

【WebGPU Unleashed】1.1 绘制三角形

一部2024新的WebGPU教程,作者Shi Yan。内容很好,翻译过来与大家共享,内容上会有改动,加上自己的理解。更多精彩内容尽在 dt.sim3d.cn ,关注公众号【sky的数孪技术】,技术交流、源码下载请添加微信号:digital_twin123 在 3D 渲染领域,三角形是最基本的绘制元素。在这里,我们将学习如何绘制单个三角形。接下来我们将制作一个简单的着色器来定义三角形内的像素

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

AI基础 L9 Local Search II 局部搜索

Local Beam search 对于当前的所有k个状态,生成它们的所有可能后继状态。 检查生成的后继状态中是否有任何状态是解决方案。 如果所有后继状态都不是解决方案,则从所有后继状态中选择k个最佳状态。 当达到预设的迭代次数或满足某个终止条件时,算法停止。 — Choose k successors randomly, biased towards good ones — Close

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动