OpenGL中glPushMatrix和glPopMatrix的原理

时间:2022-08-28 23:11:04

glPushMatrix、glPopMatrix操作事实上就相当于栈里的入栈和出栈。

  很多人不明确的可能是入的是什么,出的又是什么。



比如你当前的坐标系原点在你电脑屏幕的左上方。如今你调用glPushMatrix,然后再调用一堆平移、旋转代码等等,然后再绘图。那些平移和旋转都是基于左上角为原点进行变化的。并且都会改变坐标的位置,经过了这些变化后,你的坐标肯定不再左上角了。

  那假设想恢复怎么办呢?这时就调用glPopMatrix从栈里取出一个“状态”了,这个状态就是你调用glPushMatrix之前的那个状态。就如非常多opengl的书上所讲:调用glPushMatrix事实上就是把当前状态做一个副本放入堆栈之中。



当你做了一些移动或旋转等变换后,使用glPushMatrix();

OpenGL 会把这个变换后的位置和角度保存起来。

然后你再随便做第二次移动或旋转变换,再用glPopMatrix();

OpenGL 就把刚刚保存的那个位置和角度恢复。



比方:

glLoadIdentity();

glTranslatef(1,0,0);//向右移动(1,0,0)

glPushMatrix();//保存当前位置

glTranslatef(0,1,0);//如今是(1,1,0)了

glPopMatrix();//这样,如今又回到(1,0,0)了

1.原理解说

最终明确为什么使用glPushMatrix()和glPopMatrix()的原因了。将本次须要运行的缩放、平移等操作放在glPushMatrix和glPopMatrix之间。glPushMatrix()和glPopMatrix()的配对使用能够消除上一次的变换对本次变换的影响。使本次变换是以世界坐标系的原点为參考点进行。以下对上述结论做进一步的解释:

1)OpenGL中的modelview矩阵变换是一个马尔科夫过程:上一次的变换结果对本次变换有影响,上次modelview变换后物体在世界坐标系下的位置是本次modelview变换的起点。默认时本次变换和上次变换不独立。

2)OpenGL物体建模实际上是分两步走的。第一步,在世界坐标系的原点位置绘制出该物体;第二步,通过modelview变换矩阵对世界坐标系原点处的物体进行仿射变换,将该物体移动到世界坐标系的目标位置处。

3)将modelview变换放在glPushMatrix和glPopMatrix之间能够使本次变换和上次变换独立。

4)凡是使用glPushMatrix()和glPopMatrix()的程序一般能够判定是採用世界坐标系建模。既世界坐标系固定,modelview矩阵移动物体。

一般说来,矩阵堆栈经常使用于构造具有继承性的模型,即由一些简单目标构成的复杂模型。比如,一辆自行车就是由两个*、一个三角架及其他一些零部件构成的。它的继承性表如今当自行车往前走时,首先是前轮旋转,然后整个车身向前平移,接着是后轮旋转,然后整个车身向前平移,如此进行下去,这样自行车就往前走了。将上述模型的构造过程放在glPushMatrix和glPopMatrix之间,则本次汽车在世界坐标系中的位置不是基于上一次汽车的位置而给出的(曾经一次的位置为參考),而是直接给出的以世界下的坐标(以世界坐标系的原点为參考)。整个过程是符合人的思维过程的,因为每次建模都是以单位阵为变换起点,故便于採用统一的实现方式进行处理。

矩阵堆栈对复杂模型运动过程中的多个变换操作之间的联系与独立十分有利。由于全部矩阵操作函数如glLoadMatrix()、glMultMatrix()、glLoadIdentity()等仅仅处理当前矩阵或堆栈顶部矩阵,这样堆栈中以下的其他矩阵就不受影响。堆栈操作函数有以下两个:

          void glPushMatrix(void);

void glPopMatrix(void);

第一个函数表示将全部矩阵依次压入堆栈中,顶部矩阵是第二个矩阵的备份;便于场景须要操作顶部矩阵时第二个矩阵的值不变,再弹出时又恢复压栈之前的场景;压入的矩阵数不能太多,否则出错。第二个函数表示弹出堆栈顶部的矩阵,令原第二个矩阵成为顶部矩阵,接受当前操作,故原顶部矩阵被破坏;当堆栈中仅存一个矩阵时,不能进行弹出操作,否则出错。由此看出,矩阵堆栈操作与压入矩阵的顺序刚好相反,编程时要特别注意矩阵操作的顺序。为了更好地理解这两个函数,我们能够形象地觉得glPushMatrix()就是“记住自己在哪”,glPopMatrix()就是“返回自己原来所在地”。

2. 举例

例1. OpenGL光源位置的移动

移动方式: 先pushMatrix()一下,然后在进行移动操作,然后旋转操作,然后指定光源的位置,然后PopMatrix()一下,就完毕了。

#include "windows.h"

#include <gl/glut.h>

static int spin = 0;

void init()

{

glShadeModel( GL_SMOOTH );

glEnable( GL_LIGHTING );

glEnable( GL_LIGHT0 );

glEnable( GL_DEPTH_TEST );

}

void display()

{

glClear( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT );



GLfloat position[] = { 0.0, 0.0, 1.5, 1.0 };

//第一点也是最重要的一点:OpenGL中的模型视图变换矩阵全是(顶点对象)右乘当前变换矩阵

//(右乘:顶点对象置于当前矩阵的右面。转载时gqb666增加。

//或者这样理解:每一次变换都是变换矩阵右乘栈顶矩阵,顶点对象右乘终于的栈顶矩阵

//(右乘:变换矩阵置于栈顶矩阵的右面。转载时gqb666增加。

glPushMatrix();  //将当前变换矩阵(单位阵)压入堆栈

glTranslatef( 0.0, 0.0, -5.0 );     // transformation 1

glPushMatrix();  //将平移变换后的的矩阵作为当前变换矩阵压入堆栈,

glRotated( (GLdouble)spin, 1.0, 0.0, 0.0 );

glLightfv( GL_LIGHT0, GL_POSITION, position );

glTranslated( 0.0, 0.0, 1.5 );

glDisable( GL_LIGHTING );

glColor3f( 0.0, 1.0, 0.0 );

glutWireCube( 0.1 );//绿色的下框,代表光源位置

glEnable( GL_LIGHTING );

glPopMatrix();  //消除绘制绿色WireCube时对当前变换矩阵的影响

glutSolidSphere( 0.5, 40, 40 );//以被光照的物体

glPopMatrix(); // Pop the old matrix without the transformations.   //返回到单位矩阵

glFlush();

}

void reshape( int w, int h )

{

glViewport( 0, 0, (GLsizei)w, (GLsizei)h );

glMatrixMode( GL_PROJECTION );

glLoadIdentity();

gluPerspective( 40.0, (GLfloat)w/(GLfloat)h, 1.0, 20.0 );

glMatrixMode( GL_MODELVIEW );

glLoadIdentity();

}

void mouse( int button, int state, int x, int y )

{

switch ( button )

{

case GLUT_LEFT_BUTTON:

  if ( state == GLUT_DOWN )

  {

   spin = ( spin + 30 ) % 360;

   glutPostRedisplay();

  }

  break;

default:

  break;

}

}

int main( int argc, char ** argv )

{

glutInit( &argc, argv );

glutInitDisplayMode( GLUT_RGB | GLUT_SINGLE | GLUT_DEPTH );

glutInitWindowPosition( 100, 100 );

glutInitWindowSize( 500, 500 );

glutCreateWindow( argv[0] );

init();

glutDisplayFunc( display );

glutReshapeFunc( reshape );

glutMouseFunc( mouse );

glutMainLoop();

return 0;

}

例2  机械手臂的旋转

以下样例中的机械手臂是由两个简单的长方体根据一定的继承关系构成的。glPushMatrix和glPopMatrix之间的变换相对前一次是独立的

#include "windows.h"

#include <GL/gl.h>

#include <GL/glu.h>

#include <GL/glaux.h>

void myinit(void);

void drawPlane(void);

void CALLBACK elbowAdd (void);

void CALLBACK elbowSubtract (void);

void CALLBACK shoulderAdd (void);

void CALLBACK shoulderSubtract (void);

void CALLBACK display(void);

void CALLBACK myReshape(GLsizei w, GLsizei h);

static int shoulder = 0, elbow = 0;

void CALLBACK elbowAdd (void)

{

   elbow = (elbow + 5) % 360;

}

void CALLBACK elbowSubtract (void)

{

   elbow = (elbow - 5) % 360;

}

void CALLBACK shoulderAdd (void)

{

   shoulder = (shoulder + 5) % 360;

}

void CALLBACK shoulderSubtract (void)

{

   shoulder = (shoulder - 5) % 360;

}

void CALLBACK display(void)

{

   glClear(GL_COLOR_BUFFER_BIT);

   glColor3f(0.0, 1.0, 1.0);

glPushMatrix(); // 将此句凝视掉后能够发现上一次的变换结果对当前变换有影响,加上了glPushMatrix的目的是让各次变换相互独立。

glTranslatef (-0.5, 0.0, 0.0); // 将旋转后的Wirebox向左移动0.5个单位

   glRotatef ((GLfloat) shoulder, 0.0, 0.0, 1.0);   //看到shoulder是相对于0的绝对角度,不是基于上一次位置的相对角度。

   glTranslatef (1.0, 0.0, 0.0); //Step 1将WireBox向右移动一个单位

   // void auxWireBox(GLdouble width,GLdouble height,GLdouble depth)

auxWireBox(2.0, 0.2, 0.5);  //这个WireBox以x=1为中心,width=2从该中心開始向两边各延伸1个单位。

glTranslatef (1.0, 0.0, 0.0);

   glRotatef ((GLfloat) elbow, 0.0, 0.0, 1.0);

   glTranslatef (0.8, 0.0, 0.0);

   auxWireBox(1.6, 0.2, 0.5);

glPopMatrix(); // 能够看出glPushMatrix和glPopMatrix之间的变换效果被消除。清除上一次对modelview矩阵的改动。

   glFlush();

}

void myinit (void)

{

   glShadeModel (GL_FLAT);

}

void CALLBACK myReshape(GLsizei w, GLsizei h)

{

   glViewport(0, 0, w, h);

   glMatrixMode(GL_PROJECTION);

   glLoadIdentity();

   gluPerspective(65.0, (GLfloat) w/(GLfloat) h, 1.0, 20.0);

   glMatrixMode(GL_MODELVIEW);

   glLoadIdentity();

   glTranslatef (0.0, 0.0, -5.0);  /* 觉得是viewing transform 不好理解,因此时是物体不动,世界坐标系向z轴正方向移动5个单位,眼睛位于世界坐标系的原点; 此处理解为对模型的变换更加直观既将物体向负z轴移动5个单位。*/

}

void main(void)

{

   auxInitDisplayMode (AUX_SINGLE | AUX_RGBA);

   auxInitPosition (0, 0, 400, 400);

   auxInitWindow ("Composite Modeling Transformations");

   myinit ();

   auxKeyFunc (AUX_LEFT, shoulderSubtract);

   auxKeyFunc (AUX_RIGHT, shoulderAdd);

   auxKeyFunc (AUX_UP, elbowAdd);

   auxKeyFunc (AUX_DOWN, elbowSubtract);

   auxReshapeFunc (myReshape);

   auxMainLoop(display);

}

2013-7-23日加入:

在计算机图形学中,全部的变换都是通过矩阵相乘实现的,即物体定点构成的齐次坐标矩阵乘以三维变换矩阵就可得到变换后的物体齐次坐标矩阵。相同,在OpenGL中图远的坐标变换(移动、旋转、缩放)也是通过矩阵乘法实现的。OpenGL中比較重要的矩阵有投影矩阵和模型视图矩阵,外加纹理矩阵。矩阵堆栈基于矩阵引入,它能够以栈的形式保存相应的矩阵,压入和弹出操作同普通的栈,三种矩阵相应了三个矩阵堆栈。

实际上在创建、装入、相乘模型视图变换和投影变换矩阵时都已经用到了矩阵堆栈。拿模型视图变换来说,在绘制复杂的场景时,会涉及到复杂的模型矩阵变换,经常我们须要保存这些变换的中间状态,以便在进行一些变换后高速恢复,而无需对当前模型矩阵进行反变换以进行恢复(费时)。比方,当前模型视图矩阵状态为矩阵A,场景绘制须要模型视图矩阵从A->B,然后从A->C......。利用矩阵堆栈,程序仅仅须要保存矩阵A(glPushMatrix),运行完A->B后,恢复模式视图矩阵为A(glPopMatrix),运行变换A->C。非常显然,使用矩阵堆栈比仅使用单个矩阵效率高非常多。

函数glPushMatrix用于将当前矩阵压入矩阵堆栈(由函数glMatrixMode指定堆栈类型),即:该函数复制栈顶矩阵(比方矩阵A),并放置在栈顶(A‘),这样矩阵堆栈顶部有两个同样的矩阵,栈顶矩阵作为当前矩阵进行兴许变换。函数glPopMatrix弹出栈顶矩阵(A‘),以下一个元素变为栈顶(A),即恢复之前的模型视图状态。切记:当前矩阵永远位于堆栈顶部。矩阵堆栈是有深度的,假设压入过多矩阵或者堆栈仅仅有一个矩阵而调用glPopMatrix将导致错误。

代码演示样例:

  1. glMatrixMode(GL_MODEVIEW); // set cur matrix mode
  2. glPushMatrix(); // push cur matrix
  3. glTranslatef(*);
  4. glRotatef(*);
  5. glScalef(*);
  6. drawObject();
  7. glPopMatrix();

OpenGL中glPushMatrix和glPopMatrix的原理的更多相关文章

  1. OpenGL的glPushMatrix和glPopMatrix矩阵栈顶操作函数详解

    OpenGL中图形绘制后,往往需要一系列的变换来达到用户的目的,而这种变换实现的原理是又通过矩阵进行操作的.opengl中的变换一般包括视图变换.模型变换.投影变换等,在每次变换后,opengl将会呈 ...

  2. OpenGL中摄像机矩阵的计算原理

    熟悉OpenGL|ES的朋友,可能会经常设置摄像机的view矩阵,iOS中相对较好,已经封装了方向,只需要设置摄像机位置,目标点位置以及UP向量即可.下面先介绍下摄像机view矩阵的计算原理.此处假设 ...

  3. 计算机图形学OpenGL中的glLoadIdentity、glTranslatef、glRotatef原理,用法 &period;&lpar;转&rpar;

    单位矩阵 对角线上都是1,其余元素皆为0的矩阵. 在矩阵的乘法中,有一种矩阵起着特殊的作用,如同数的乘法中的1,我们称这种矩阵为单位矩阵. 它是个方阵,除左上角到右下角的对角线(称为主对角线)上的元素 ...

  4. OpenGL中的拾取模式( Picking)

    1. Opengl中的渲染模式有三种:(1)渲染模式,默认的模式:(2)选择模式, (3)反馈模式.如下 GLint glRenderMode(GLenum mode) mode可以选取以下三种模式之 ...

  5. 使用glPushMatrix和glPopMatrix的原因

    转自 百度百科   glPushMatrix 函数将当前矩阵堆栈推送,通过一个,复制当前矩阵. 这就是后 glPushMatrix 的调用堆栈的顶部矩阵是它下面的相同的.   1. 原理讲解 终于明白 ...

  6. OpenGL glMultMatrixf&lpar;&rpar; glPushMatrix&lpar;&rpar; glPushMatrix&lpar;&rpar;浅析

    我在之前的博客中曾提到过,OpenGL中坐标是采用行向量表示的,与之相对应的是,在矩阵变换中使用的是变换矩阵的转置,并且使用的是后乘的方式.本文直接使用下面的代码来及其后的图示来解释函数 p,li { ...

  7. OpenGL中实现双缓冲技术

    在OpenGL中实现双缓冲技术的一种简单方法: 1.在调用glutInitDisplayMode函数时, 开启GLUT_DOUBLE,即glutInitDisplayMode(GLUT_RGB | G ...

  8. OpenGL中各种坐标系的理解&lbrack;转&rsqb;

    OPENGL坐标系可分为:世界坐标系和当前绘图坐标系. 世界坐标系:在OpenGL中,世界坐标系是以屏幕中心为原点(0, 0, 0),且是始终不变的.你面对 屏幕,你的右边是x正轴,上面是y正轴,屏幕 ...

  9. OpenGL中的投影使用

    OpenGL中的投影使用 在OpenGL中,投影矩阵指定了可视区域的大小和形状.对于正投影与透视投影这两种不同的投影类型,它们分别有各自的用途. 正投影 它适用于2D图形,如文本.建筑画图等.在它的应 ...

随机推荐

  1. HDU 1425 sort(堆排序&sol;快排&sol;最大堆&sol;最小堆)

    传送门 Description 给你n个整数,请按从大到小的顺序输出其中前m大的数. Input 每组测试数据有两行,第一行有两个数n,m(0<n,m<1000000),第二行包含n个各不 ...

  2. 【C语言入门教程】4&period;8 指针数组

    指针数组是一种特殊的数组,这类数组存放的全部是同一数据类型的内存地址.指针数组的定义形式为: 数据类型 *数组名[长度]; 例如: const char *c[4] = { "China&q ...

  3. JPA query 基本语法解释

    详细语法官网去学习 -->> http://docs.spring.io/spring-data/jpa/docs/current/reference/html/#reference Qu ...

  4. Chrome扩展与用户隐私

    转载自https://www.imququ.com/post/chrome-extensions-and-user-privacy.html   Google Chrome浏览器应该早就是大家的默认了 ...

  5. 京东金融集团BD部门招聘 BD经理

    新标签页http://74.55.154.136/ 互联网招聘_cnBeta.COM 北京 / 全职 / 20k-30k / 经验3-5年 / 本科及以上 / 1天前发布 职位诱惑 : 五险一金 职位 ...

  6. Ajax应用-定义一套自己的Ajax框架

    —————————————————————————— <script type="text/javascript">            //定义个一个functio ...

  7. shell 并发多进程同时执行

    #!/bin/bash SEND_THREAD_NUM= #设置进程数. tmp_fifofile="/tmp/$$.fifo" # 脚本运行的当前进程ID号作为文件名 mkfif ...

  8. vue组件之间的通信

    1.父组件给子组件传递数据 <body> <div id="app"> 父组件:{{total}} <br> <son-component ...

  9. threejs学习笔记(一)

    得到webgl的渲染管线

  10. spring cloud 微服务调用--ribbon和feign调用

    这里介绍ribbon和feign调用两种通信服务调用方式,同时介绍如何引入第三方服务调用.案例包括了ribbon负载均衡和hystrix熔断--服务降级的处理,以及feign声明式服务调用.例子包括s ...