图形图像必须知道的事(一):几何变换

作者:mznewfacer(Wolf Geek)     时间:20111130日         欢迎转载     ,请注明出处!

    这两天又不能安生了,得写论文,好吧,先把arduino断下来,似乎我同时只能干一件事,严重怀疑自己的智商!顺便写一些基础总结吧,已备后用。可能会比较枯燥,但是必须强调的是这些都是图形图像方面的数学基础,不可忽视啊!

    先看几个变换公式:



至此,二维平面上的所有变换集合就已叙述完毕,具体形式如下图所示。


对于三维图像而言,对应的变换矩阵就变为四阶矩阵。




这就是初始化坐标系,对应的4阶齐次矩阵为四阶单位阵。


在此坐标系上画点[0.5 ; 0.5 ; 0.5],之后在Oxy平面上做投影,所得结果如下图所示:


此时变换矩阵为


之后将坐标系绕z轴旋转45度并且在y轴方向上平移1.5个刻度,变换矩阵为


最后连续绕x轴,y轴,z轴分别旋转45度,得到结果如下图所示:


变换矩阵可以由此推得:


至于为什么引入齐次坐标系,这里就不做多余解释,详情可见这里

最后用opengl画一下类似的坐标系变换,

#include <GL/glut.h>

float w, h, tip = 0, turn = 0;

float ORG[3] = {0,0,0};

float XP[3] = {1,0,0}, XN[3] = {-1,0,0},
YP[3] = {0,1,0}, YN[3] = {0,-1,0},
ZP[3] = {0,0,1}, ZN[3] = {0,0,-1},VP[3]={0, 0, -5};

void reshape (int nw, int nh)
{
	w = nw;
	h = nh;
}

void Turn (int key, int x, int y)
{
	switch (key) {
case GLUT_KEY_RIGHT: turn += 5; break;
case GLUT_KEY_LEFT : turn -= 5; break;
case GLUT_KEY_UP : tip -= 5; break;
case GLUT_KEY_DOWN : tip += 5; break;
	}
}

void Draw_Axes (void)
{
	glPushMatrix ();

	glTranslatef (-1.5, -1.5, -5);
	glRotatef (tip , 1,0,0);
	glRotatef (turn, 0,1,0);
	glScalef (1, 1, 1);

	glLineWidth (2.0);

	glBegin (GL_LINES);
	glColor3f (1,0,0); // X axis is red.
	glVertex3fv (ORG);
	glVertex3fv (XP ); 
	glColor3f (0,1,0); // Y axis is green.
	glVertex3fv (ORG);
	glVertex3fv (YP );
	glColor3f (0,0,1); // z axis is blue.
	glVertex3fv (ORG);
	glVertex3fv (ZP ); 
	glEnd();

	glTranslatef (1, 1, 1);
	glBegin (GL_LINES);
	glColor3f (1,0,0); // X axis is red.
	glVertex3fv (ORG);
	glVertex3fv (XN ); 
	glColor3f (0,1,0); // Y axis is green.
	glVertex3fv (ORG);
	glVertex3fv (YN );
	glColor3f (0,0,1); // z axis is blue.
	glVertex3fv (ORG);
	glVertex3fv (ZN ); 
	glEnd();
	glTranslatef (-1, 0, 0);
	glRotatef (-90, 0,1,0);
	glBegin (GL_LINES);
	glColor3f (0,0,1); // X axis is blue.
	glVertex3fv (ORG);
	glVertex3fv (XN ); 
	glColor3f (0,1,0); // Y axis is green.
	glVertex3fv (ORG);
	glVertex3fv (YN );
	glColor3f (1,0,0); // z axis is red
	glVertex3fv (ORG);
	glVertex3fv (ZN); 
	glEnd();
	glPointSize(4.0f);
	glRotatef (-45, 0,1,0);
	glBegin(GL_POINTS);
	glColor3f (1,0,0); // X axis is red.
	glVertex3f(-1.0f, -1.0f, 0.0f);
	glEnd();
	glRotatef (-90, 0,1,0);
	glBegin(GL_POINTS);
	glColor3f (1,0,0); 
	glVertex3f(1.0f, -1.0f, 0.0f);
	glEnd();
	glLoadIdentity();
	glBegin(GL_POINTS);
	glColor3f (1,0,0);
	glVertex3fv(VP);
	glEnd();
	glPopMatrix ();
}


void display (void)
{
	glViewport (0, 0, w, h);
	glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
	Draw_Axes ();
	glutSwapBuffers ();
}

void main (void)
{
	glutInitWindowSize (600, 400);
	glutInitWindowPosition (400, 300);
	glutInitDisplayMode (GLUT_DEPTH | GLUT_DOUBLE | GLUT_RGB);
	glutCreateWindow ("坐标系变换");
	glutDisplayFunc (display);
	glutIdleFunc (display);
	glutReshapeFunc (reshape);
	glutSpecialFunc (Turn);

	glClearColor (0.1, 0.2, 0.1, 1.0);
	glEnable (GL_DEPTH_TEST);
	glMatrixMode (GL_PROJECTION);
	gluPerspective (50.0, 1.5, 1.0, 10.0);
	glMatrixMode (GL_MODELVIEW);

	glutMainLoop ();
}
效果图

    至于opengl的具体坐标系转换,像是世界坐标系到观察坐标系的转换,如何利用GL_PROJECTION投影矩阵将3维投影到2维上,投影时选择glFrustum()还是glOrtho(),以及各个变换的具体原理, Song Ho Ahn 的博客中已经写得相当详细,里面有两个例子也很不错。理论部分例子部分


  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值