之前在学习Android的时候有写过如果在Android中使用OpenGL,当时完全都是用java语言来实现的,现在我们用NDK来实现一次。

     实现的思路就是将渲染器中的onDrawFrame,onSurfaceChanged,onSurfaceCreated分别在C中实现,然后将C编译成.so文件之后在Java中直接调用相应的函数就可以了。

     步骤就不详细叙述了,代码贴一下。

    主Activity:

package com.empty.ndkgl;  import com.example.ndkgl.R;  import android.opengl.GLSurfaceView; import android.os.Bundle; import android.app.Activity; import android.view.Menu;  public class NdkGlActivity extends Activity {  	@Override 	protected void onCreate(Bundle savedInstanceState) { 		super.onCreate(savedInstanceState); 		GLSurfaceView surface = new GLSurfaceView(this);           surface.setRenderer(new NdkGlRender());           setContentView(surface);  	} 	static {           //load library            System.loadLibrary("NdkGLRenderer");      }   	@Override 	public boolean onCreateOptionsMenu(Menu menu) { 		// Inflate the menu; this adds items to the action bar if it is present. 		getMenuInflater().inflate(R.menu.activity_ndkgl, menu); 		return true; 	}  } 

Render类代码:

package com.empty.ndkgl;  import javax.microedition.khronos.egl.EGLConfig; import javax.microedition.khronos.opengles.GL10;  import android.opengl.GLSurfaceView.Renderer;  public class NdkGlRender implements Renderer{  	//declare native function 	 native private void onNdkSurfaceCreated ();   	 native private void onNdkSurfaceChanged (int width, int height);   	 native private void onNdkDrawFrame(); 	@Override 	public void onDrawFrame(GL10 arg0) { 		// TODO Auto-generated method stub 		onNdkDrawFrame ();  	}  	@Override 	public void onSurfaceChanged(GL10 gl, int width, int height) { 		// TODO Auto-generated method stub 		onNdkSurfaceChanged (width, height); 	}  	@Override 	public void onSurfaceCreated(GL10 gl, EGLConfig config) { 		// TODO Auto-generated method stub 		onNdkSurfaceCreated (); 	} 	  } 

在工程目录下创建jni文件夹,用下面的命令生成.h文件。

javah -classpath bin/classes -d jni com.empty.ndkgl.NdkGlRender 

根据头文件来创建.c文件。

注:虽然生产的 .h文件在编译的时候并没有什么作用,但还是建议做这一步,因为.c文件中的函数名一定要和.h文件中的函数名一致,最后的程序才能正常运行,不然会出现如

java.lang.UnsatisfiedLinkError的bug。

#include <jni.h> #include <GLES/gl.h> unsigned int vbo[2]; float positions[12] = {1,-1,0, 1,1,0, -1,-1,0, -1,1,0}; short indices  [4]  = {0,1,2,3}; JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkSurfaceCreated (JNIEnv* env, jobject obj) { 	//生成两个缓存区对象 	glGenBuffers (2, vbo); 	//绑定第一个缓存对象 	glBindBuffer (GL_ARRAY_BUFFER, vbo[0]); 	//创建和初始化第一个缓存区对象的数据 	glBufferData (GL_ARRAY_BUFFER, 4*12, positions, GL_STATIC_DRAW); 	//绑定第二个缓存对象 	glBindBuffer (GL_ELEMENT_ARRAY_BUFFER, vbo[1]); 	//创建和初始化第二个缓存区对象的数据 	glBufferData (GL_ELEMENT_ARRAY_BUFFER, 2*4, indices, GL_STATIC_DRAW); } JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkSurfaceChanged(JNIEnv* env, jobject obj, jint width, jint height) { 	//图形最终显示到屏幕的区域的位置、长和宽 	glViewport (0,0,width,height); 	//指定矩阵 	glMatrixMode   (GL_PROJECTION); 	//将当前的矩阵设置为glMatrixMode指定的矩阵 	glLoadIdentity (); 	glOrthof(-2, 2, -2, 2, -2, 2); }  JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkDrawFrame (JNIEnv* env, jobject obj) { 	//启用顶点设置功能,之后必须要关闭功能 	glEnableClientState (GL_VERTEX_ARRAY); 	//清屏 	glClearColor (0,0,1,1); 	glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); 	glMatrixMode (GL_MODELVIEW); 	glLoadIdentity (); 	glBindBuffer    (GL_ARRAY_BUFFER, vbo[0]); 	//定义顶点坐标 	glVertexPointer (3, GL_FLOAT, 0, 0); 	glBindBuffer    (GL_ELEMENT_ARRAY_BUFFER, vbo[1]); 	//按照参数给定的值绘制图形 	glDrawElements  (GL_TRIANGLE_STRIP, 4, GL_UNSIGNED_SHORT, 0); 	//关闭顶点设置功能 	glDisableClientState(GL_VERTEX_ARRAY); } 

编写Android.mk

#FileName:Android.mk #Description:makefile of NdkGl LOCAL_PATH := $(call my-dir)  include $(CLEAR_VARS)  LOCAL_MODULE    := NdkGLRenderer LOCAL_SRC_FILES := com_empty_ndkgl_NdkGlRender.c   LOCAL_LDLIBS := -lGLESv1_CM  include $(BUILD_SHARED_LIBRARY) 

编译库

$NDK_ROOT/ndk-build

在Eclipse中运行程序



通过修改.c的实现,我们就可以绘制不同的图形。

比如网格圆球:

#include <jni.h> #include <unistd.h> #include <stdlib.h> #include <stdio.h> #include <math.h> #include <GLES/gl.h> typedef unsigned char byte; typedef struct {    GLfloat x,y,z; } XYZ; float rotateQuad; #define PI 3.14159265 #define DTOR PI/180 static byte indices[8]={0,1,1,2,2,3,3,0};  //索引数组  void CreateUnitSphere(int dtheta,int dphi) {    int n;    int theta,phi;    XYZ p[4];     for (theta=-90;theta<=90-dtheta;theta+=dtheta) {       for (phi=0;phi<=360-dphi;phi+=dphi) {          n = 0;          p[n].x = cos(theta*DTOR) * cos(phi*DTOR);          p[n].y = cos(theta*DTOR) * sin(phi*DTOR);          p[n].z = sin(theta*DTOR);          n++;          p[n].x = cos((theta+dtheta)*DTOR) * cos(phi*DTOR);          p[n].y = cos((theta+dtheta)*DTOR) * sin(phi*DTOR);          p[n].z = sin((theta+dtheta)*DTOR);          n++;          p[n].x = cos((theta+dtheta)*DTOR) * cos((phi+dphi)*DTOR);          p[n].y = cos((theta+dtheta)*DTOR) * sin((phi+dphi)*DTOR);          p[n].z = sin((theta+dtheta)*DTOR);          n++;          if (theta >=-90 && theta <= 90) {             p[n].x = cos(theta*DTOR) * cos((phi+dphi)*DTOR);             p[n].y = cos(theta*DTOR) * sin((phi+dphi)*DTOR);             p[n].z = sin(theta*DTOR);             n++;          }           /* Do something with the n vertex facet p */  		glVertexPointer(3, GL_FLOAT, 0, p); 		glDrawElements(GL_LINES, 8, GL_UNSIGNED_BYTE, indices);         }    } }   JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkSurfaceCreated (JNIEnv* env, jobject obj) { 		// 启用阴影平滑 	glShadeModel(GL_SMOOTH);  	// 黑色背景 	glClearColor(0, 0, 0, 0);  	// 设置深度缓存 	glClearDepthf(1.0f); 	// 启用深度测试 	glEnable(GL_DEPTH_TEST); 	// 所作深度测试的类型 	glDepthFunc(GL_LEQUAL);  	// 告诉系统对透视进行修正 	glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_FASTEST);} JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkSurfaceChanged(JNIEnv* env, jobject obj, jint width, jint height) { 	//图形最终显示到屏幕的区域的位置、长和宽 	glViewport (0,0,width,height); 	//指定矩阵 	glMatrixMode   (GL_PROJECTION); 	//将当前的矩阵设置为glMatrixMode指定的矩阵 	glLoadIdentity (); 	glOrthof(-2, 2, -2, 2, -2, 2); }  JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkDrawFrame (JNIEnv* env, jobject obj) { 	 //启用顶点设置功能,之后必须要关闭功能     glEnableClientState (GL_VERTEX_ARRAY); 	 //清屏     glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);      glMatrixMode (GL_MODELVIEW);     glLoadIdentity ();  	glFrontFace(GL_CW);  	glRotatef(rotateQuad, 1.0f, 1.0f, 0.0f);//旋转效果     CreateUnitSphere(10,10); 	 //关闭顶点设置功能 	 glDisableClientState(GL_VERTEX_ARRAY); 	rotateQuad -= 1.5f; }



立方体

#include <jni.h> #include <unistd.h> #include <stdlib.h> #include <stdio.h> #include <math.h> #include <GLES/gl.h> #define col 1.0f #define pos 1.0f #define PI 3.14159265 static GLfloat vertex[] = {                -pos,-pos,-pos,	/*0*/ 				-pos,-pos,pos,	/*1*/ 				pos,-pos,pos,	/*2*/ 				pos,-pos,-pos,	/*3*/ 				-pos,pos,-pos,	/*4*/ 				-pos,pos,pos,	/*5*/ 				pos,pos,pos,	/*6*/ 				pos,pos,-pos,	/*7*/        };         static GLfloat colors[] = {                col,0,0,col, 				0,col,0,col, 				0,0,col,col, 				col,col,0,col, 				col,0,col,col, 				0,col,col,col, 				0,0,0,col, 				col,col,col,col,        };         static GLubyte mindex[] = {                 0,2,1,	0,3,2, 				5,1,6,	6,1,2, 				6,2,7,	7,2,3, 				0,4,3,	4,7,3, 				4,0,1,	4,1,5, 				4,5,6,	4,6,7,        };  static GLfloat angle = 0.0f;   JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkSurfaceCreated (JNIEnv* env, jobject obj) { 	// 启用阴影平滑 	glShadeModel(GL_SMOOTH);  	// 黑色背景 	glClearColor(0, 0, 0, 0);  	// 设置深度缓存 	glClearDepthf(1.0f); 	// 启用深度测试 	glEnable(GL_DEPTH_TEST); 	// 所作深度测试的类型 	glDepthFunc(GL_LEQUAL); 	// 告诉系统对透视进行修正 	glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_FASTEST); } static void _gluPerspective(GLfloat fovy, GLfloat aspect, GLfloat zNear, GLfloat zFar)   {       GLfloat top = zNear * ((GLfloat) tan(fovy * PI / 360.0));       GLfloat bottom = -top;       GLfloat left = bottom * aspect;       GLfloat right = top * aspect;       glFrustumf(left, right, bottom, top, zNear, zFar);   } JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkSurfaceChanged(JNIEnv* env, jobject obj, jint width, jint height) { 	if (height==0)                              // 防止被零除         {                 height=1;                         // 将Height设为1         }            glViewport(0, 0, width, height);                  // 重置当前的视口         glMatrixMode(GL_PROJECTION);                      // 选择投影矩阵         glLoadIdentity();                         // 重置投影矩阵            GLfloat ratio = (GLfloat)width/(GLfloat)height;         // 设置视口的大小         _gluPerspective(45.0f,(GLfloat)width/(GLfloat)height,0.1f,100.0f);     //    glOrthof(-2.0f, 2.0f, -2.0f, 2.0f, -2.0f, 2.0f);            glMatrixMode(GL_MODELVIEW);                       // 选择模型观察矩阵         glLoadIdentity();                         // 重置模型观察矩阵 }  JNIEXPORT void JNICALL Java_com_empty_ndkgl_NdkGlRender_onNdkDrawFrame (JNIEnv* env, jobject obj) { 	glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);         	glMatrixMode(GL_MODELVIEW);   	glLoadIdentity();   	glTranslatef(0, 0, -8.0f);  	glRotatef(angle, 0, 1.0F, 0);  	glRotatef(angle, 0, 0, 1.0F); 		   	glEnableClientState(GL_VERTEX_ARRAY);  	glEnableClientState(GL_COLOR_ARRAY);             	glVertexPointer(3,GL_FLOAT,0,vertex);  	glColorPointer(4,GL_FLOAT,0,colors);             	glDrawElements(GL_TRIANGLES,36,GL_UNSIGNED_BYTE,mindex);         	glDisableClientState(GL_VERTEX_ARRAY);     glDisableClientState(GL_COLOR_ARRAY);     angle += 1.2f; }


打完收工。


2013.1.15日更新:

发现一个更强的demo,就在NDK的samples文件夹中,san-angeles就是了!

San Angeles Observation,XX大赛的冠军,原程序只有4K,被google收录到NDK的demo里面了,下面我们就跑一下它。

直接在Eclipse中创建Android工程,选择Android Project From Exiting Code。


直接跑的话会报错,提示无法初始化,我们必须先把c编译成.so.

命令行进入到项目文件夹,执行ndk-build



再修改一下Activity,让它全屏幕现实,只修改onCreate函数就可以了。

    @Override     protected void onCreate(Bundle savedInstanceState) {         super.onCreate(savedInstanceState);         mGLView = new DemoGLSurfaceView(this);         requestWindowFeature(Window.FEATURE_NO_TITLE);           getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,                            WindowManager.LayoutParams.FLAG_FULLSCREEN);         setContentView(mGLView);     }

运行结果: