| 网站首页 | 文秘范文 | 论文中心 | 小品剧本 | 小说 | 藏金阁 | 留言簿 | 汇款帮助 | 客服中心 | FAQ | 电视 | 免费文秘 | 代写 | 
您现在的位置: 中国文秘网 >> 论文中心 >> 计算机类论文 >> 正文 用户登录 新用户注册
→中国文秘网温馨提示:为方便你访问本站,请将本站设为首页或加入收藏夹中(点击加入收藏)。
紧急公告:近来发现有些不法网站复制本站版面进行欺骗,为防止上当,敬请会员记好本站网址或把本站加入收藏夹中。
轻松入会,年卡、点卡任君选择,QQ及电话24小时服务,付款后5分钟开通,在线QQ:87651921 ,客服电话:013923833528,详情见"汇款须知"
用计算机程序制作三维立体画          【字体:
用计算机程序制作三维立体画
作者:佚名    论文来源:中国文秘网    点击数:1514    更新时间:2006-4-5
将此页收藏到: 网摘中国 | 新浪 | 热门 | Hao6 | 和讯 | 天极 | YouNote | 5Seek | 365Fav | 365key |博采 | 亿友响享 | 狐摘
3万篇免费论文,近200个详细分类,为你的论文写作排忧解难。点击进入 摘 要 该文介绍了三维立体画的原理和制作方法,并给出了用C语言编写的源程序。借助于Pbrush.exe,读者可以自己设计和欣赏各式各样的三维立体画。
目前,市面上正在流行各式各样的立体画,其特点是从外表来看与一般的图案很相似,但是双眼紧盯着注视片刻后,一恍惚之间眼前便出现了画中画——立体像。笔者第一次看到这种画便被发明者的创意所倾倒。利用众所皆知的双眼视差原理,竟能在一张平面纸上制造出如此奇幻。但是立体画本身除了其发明者的灵感和画面创作者的别出心裁之外,其原理上并无神秘之处。用计算机程序来实现它,可说是易如反掌。笔者用一个晚上时间,便在微机上用BASIC语言实现了简单形体——平面圆饼的立体画。当然,要使该程序具有完善的功能,提高其制作速度,还是应该用编译语言(如C语言)来编写。本文中给出的源程序借助于Windows中的.BMP图形文件,可使大家自己制作任意形态的立体画。
一、立体画的原理
看过立体电影的人都知道,当人的双眼分别接收不同视角拍摄的图像时便会产生立体感。这是由于人眼长期观察的习惯造成的。和立体电影原理相同的立体摄影风景照片也很早就已出现。图1中给出了这种立体照片的示意图。左、右照片分别是人的双眼角度上观察一棱锥体时左右眼看到的图像(图2)。左眼看到的是棱锥的顶端向右错动了一些的图像,右眼的看到则是棱锥的顶端向左错动了一些的图像。如果用一张硬卡片隔开两张照片(如图3),@@09A04000.GIF;图1@@
@@09A04001.GIF;图2@@
@@09A04002.GIF;图3双眼分别看两张画,会看到一个立体的棱锥体。这种立体照片的观察方法在测绘学中也早已采用。
但是,目前的三维立体画在形式上与这些很不相同。它是怎样在同一张画面上呈现立体的呢?首先,分析一下人们是怎样从这些立体画中看出“立体形体”的。从前面所说的可以知道,人眼要得到立体感,双眼必须有视差,即双眼看到的图像应该有差异。人们在看立体画时,都有“恍惚”一下的过程。在这过程中,双眼的视中心发生了错动(如图4)。这样@@09A04003.GIF;图4左眼看到的是画面的“偏左像”,右眼看到的是画面的“偏右像”。@@只要“偏左像”和“偏右像”的内容相当于图1的左、右照片,双眼就会感到立体形体。那么,能否把图1的左、右照片分别当做“偏左像”和“偏右像”,简单重叠来得到立体画呢?
显然不行。能够合成立体画的“偏左像”和“偏右像”是要满足一定条件的。
如果图5中表现的棱锥体的表面上有图案的话,
@@09A04004.GIF;图5像素a和像素a'应该具有相同的颜色,因为它们是从不同视角观察的@@同一个实体点。像素b和像素b'、像素c和像素c'的情况与此相同。把两幅画分别当作“偏左图”和“偏右图”,部分重叠成为同一画面时,在新的画面上这种关系仍应该表现为a=a',b=b',c=c'(如图6)。但这时应该注意到,在这张合成
@@09A04005.GIF;图6画面上,点a'既是“偏右图”上的点a',又是“偏左图”上的点b。而@@一张画面上相同坐标点的像素只可能是一种颜色,因此,产生了新的像素关系a'=b。另外,点a既是“偏左图”上的点a,又是“偏右图”上的点c',所以,a=c'。以此类推,点b'和点c也有类似的情况。因此出现了新的关系表示式,...'=c=c'=a=a'=b=b'=...。这就构成了立体画面上像素必须要满足的条件:“等颜色像素链”。立体画上的所有点都从属于某一条“等颜色像素链”。这就是所有立体画图案都呈现出某种程度上的水平周期性的原因。
因此,对于任意立体形状,只要构造出相应的这种“等像素链”,并按其规律充填图案即可得到立体画。但是正如前面所述,由于这种“等像素链”条件的约束,人们虽然可以随意构造出各种形体的立体画,但其立体形体的表面图案是不能完全随人意愿的。
二、制作立体画的计算机程序
由于人的双眼的水平性,以上的“等像素链”只按水平方向分布,与垂直方向无关。因此,在程序中,各个像素行的处理过程是相互独立的。制作立体画的程序主结构图如图7。
@@09A04006.GIF;图7 制作立体画的程序主结构图在以上结构图中,关键是如何建立“等@@像素链”。具体的处理如下。对于立体形体上的每一个点,首先求出该点在“偏左图”和“偏右图”上的坐标。以图1中的棱锥顶点为例,实际上其X坐标是在中心点,但由于双眼的位置并不在其正上方,顶点在“偏左图”上向右位移,在“偏右图”上向左位移,而且其位移值的大小显然与其高度有关,即该点坐标越高位移值就越大。a,b,c等点也都有这些位移。在求出一个点在“偏左图”和“偏右图”上的坐标后,再算出在合成图(如图6)上的对应坐标
,以建立“等像素”关系,如a=a'。当立体形体的一个水平剖面上的全部点经过以上处理后,合成图的各条“等像素链”关系也就自然形成了。
另外,由于有可能出现高点遮盖低点的情况,“等像素链”的构造应该从低点到高点逐层进行,高点的“等像素”关系将替代低点的“等像素”关系。这也是程序主结构图中“首先,对于没有任何形体存在的背景平面构造‘等像素链’”的原因。
下面给出了根据以上结构图用C语言编写的源程序。程序中,每一个坐标点对应一个结构型数据,它包含“前像素”、“后像素”两个指针。“前像素”指针指向该坐标点作为“偏右图”上的一点,在“偏左图”所对应的点的坐标。“后像素”指针指向该坐标点作为“偏左图”上的一点,在“偏右图”所对应的点的坐标。程序中,“立体形体水平剖面的高低坐标数据”、“原始图案素材”和输出的“立体画”的文件格式都是采用了Windows3.1 的Pbrush产生的BMP图形文件格式。图幅大小要求都是640×400,用16种颜色方式。其中,立体形体上各点的高低坐标用图形文件中的颜色值表示,因此该图形文件的图形与带颜色的等高线图安全相同。通常情况下,在16色的BMP文件中颜色值从小到大的顺序为:黑色、暗红色、暗绿色、暗黄色、暗蓝色、暗紫色、暗青色、暗灰色、灰色、明红色、明绿色、明黄色、明蓝色、明紫色、明青色、白色。本程序采用最简单的“图案充填”方案,即各条“链”上的像素点皆采用该“链”上的第一个像素的颜色。程序中的常数EYE-SPACE表示“偏左图”和“偏右图”之间的偏差,BO-DOT是表明“链”的首或尾的指针标志。
该程序寄生在Windows 3.1中的Pbrush软件上。借助于它来构筑立体形体(即立体形体水平剖面高低坐标数据文件图8),设计原始图案(图9)。程序运行后,逐行输入并处理以上两个文件中的图形,然后输出立体画结果文件(图10)。最后,用Pbrush来观赏立体画result.bmp。当然,要设计出令人赏心悦目的立体画,必须在立体形体和图案素材的选择和搭配上做到天衣无缝,独具匠心。
@@09A04007.GIF;图8@@
@@09A04008.GIF;图9@@
@@09A04009.GIF;图10程序清单@@
/*--from 1995.4.19--to 1995.5.18----*/include<stdio.h>
#define COMPRESSION 0
#define SIZE-OF-BITMAPFILEHEADER 14
#define SIZE-OF-BITMAPINFOHEADER 40
#define SIZE-OF-RGBQUAD 4
#define PIXEL-DATE-OFFSET 14+40+4*16
/*SIZE-OF-BITMAPFILEHEADER+SIZE-OF-BITMAPINFOHEADER+BITS-PER-PIXEL
*NUM-COLOR*/
#define NUM-COLOR 16
#define NUM-LINE 400
#define WIDTH 640
#define BITS-PER-PIXEL 4
#define PIXEL-PER-BYTE 2 /*8/BITS-PER-PIXEL*/
#define BYTE-PER-LINE 320 /*((WIDTH*BITS-PER-PIXEL-1)/32+1)*4 */
#define NO-DOT WIDTH+1
#define EYE-SPACE 128
struct{
unsignde char color ;
unsigned int pri-x ;
unsigned int nxt-x ;
}dot[WIDTH];
main()
{
struct tagBITMAPFILEHEADER{
unsigned char bfType1,bfType2; /* always equal to'BM'*/
unsigned long int bfSize; /*size of file */
unsigned int bfReserved1,bfReserved2; /* set to zero */
unsiged long int bfoffits; /*byte offset from BITMAPFILEHEADER to bitmap p
ixel
data in the file */
}BITMAPFILEHEADER;
struct tagBITMAPINFOHEADER{
unsigned long int biSize,/* size of BITMAPINFOHEADER */
biWidth;/* width in pixels biHeight;/* height in pixels */
unsigned int biPlanes, /* always 1 */
biBitCount; /* color bits per pixel must be 1,4,8 or 24 */
unsigned long int biCompression, /*BI-RGB,BI-RLE 8 or 4*/
biSizeImage, /*total bytes in image */
biXPelsPerMeter,/* 0,or opt,h res. */
biYPelsPerMeter,/* 0,or opt,h res. */
biClrUsed, /* normally 0,can set a lower no. colors than biBitCount */
biClrImportant; /* normally 0 */
}BITMAPINFOHEADER;
struct tagRGBQUAD{
unsigned char rgbBlue, /* blue intensity,0-255 */
rgbGreen, /* green intensity,0-255 */
rgbRed, /* red intensity,0-255 */
rgbReserved; /* reserved,set to Zero */
}RGBQUAD[NUM-COLOR];
char *fn-layer="layer.bmp";
char *fn-org="origin.bmp";
char *fn-result="result.bmp";
FILE *flayer,*fOrigin, *fResult;
unsigned char tmp-byte1,tmp-byte2;
unsigned int line,i-byte,i-pixel,x;
unsigned int layer;
int left-x,right-x;tmp-x;
unsigned long int cur-offset;
unsigned char h[WIDTH],org-color[WIDTH];
puts("---WINTRICK---");
puts("---by Li Jisong ---");
if( (fLayer=fopen(fn-layer,"rb") )!=NULL) {
fread(&BITMAPFILEHEADER,SIZE-OF-BITMAPFILEHADER,1,fLayer);
fread(&BITMAPINFOHEADER,SIZE-OF-BITMAPINFOHEADER,1,fLayer);
if( BITMAPFILEHEADER.bfType1=='B' && BITMAPFILEHEADER.bfType2=='M'
&& BITMAPINFOHEADER.biWidth==WIDTH && BITMAPINFOHEADER.biHeight==NUM-
LINE
&& BITMAPINFOHEADER.biBitCount==BITS-PER-PIXEL
&& BITMAPINFOHEADER.biCompression==COMPRESSION)
fread(RGBQUAD,SIZE-OF-RGBQUAD,NUM-COLOR,fLayer);
else{
fclose(fLayer);
printf("File %s is not fit for this program!\n",fn-layer);
getch();
exit(1);
}
}
else{
printf("File %s does not exist!\n",fn-layer);
getch();
exit(2);
}
if( (fOrigin=fopen(fn-org,"rb"))!=NULL) {
fread(& BITMAPFILEHEADER,SIZE-OF-BITMAPFILEHEADER,1,fOrigin);
fread(& BITMAPINFOHEADER,SIZE-OF-BITMAPINFOHEADER,1,fOrigin);
if( BITMAPFILEHEADER.bfType1=='B' && BITMAPFILEHEADER.bfType2=='M'
&& BITMAPINFOHEADER.biWidth==WIDTH && BITMAPINFOHDADER.biHeight==NUM-LINE
&& BITMAPINFOHEADER.biBitCount==BITS-PER-PIXEL
&& BITMAPINFOHEADER.biCompression==COMPRESSION)
fread(RGBQUAD,SIZE-OF-RGBQUAD,NUM-COLOR,fOrigin);
else {
fclose(fOrigin);
printf("File %s is not fit for this program!\n",fn-org);
getch();
exit(3);
}
}
else {
printf("File %s does not exist!\n",fn-org);
getch();
exit(4);
}
if( (fResult=fopen(fn-result,"wb"))!=NULL){
fwrite(&BITMAPFILEHEADER,SIZE-OF-BITMAPFILEHEADER,1,fResult);
fwrite(& BITMAPINFOHIADER,SIZE-OF-BITMAPINFOHEADER,1,fResult);
fwrite(RGBQUAD,SIZE-OF-RGBQUAD,NUM-COLOR,fResult);
}
else {
printf("File %s open error!\n",fn-result);
getch();
exit(5);
}
for(line=0;line<NUM-LINE;line++){
printf("line=%d\n",line);
cur-offset=(unsigned long int)PIXEL-DATA-OFFSET+(unsigned long int)BYTE-
PER-LINE*line;
fseek (fLayer,cur-offset,SEEK-SET);
fseek (fOrigin,cur-offset,SEEK-SET);
for(i-byte=0;i-byte<BYTE-PER-LINE;i-byte++) {
fread(&tmp-byte1,1,1,fLayer);
fread(&tmp-byte2,1,1,fOrigin);
for(i-pixel=0;i-pixel<PIXEL-PER-BYTE;i-pixel++) {
x=i-byte*PIXEL-PER-BYTE+i-pixel;
if(x<WIDTH) {
h[x]=(unsigned char) (tmp-byte1<<(BITS-PER-PIXEL*i-pixel)
) /((unsigned char)0x80>>(BITS-PER-PIXEL-1));
org-color[x]=(unsigned char) (tmp-byte2<<(BITS-PER-PIXEL*i-pixel))
/((unsigned char)0x80>>(BITS-PER-PIXEL-1));
}
}}
for(x=0;x<WIDTH;x++) {
dot[x].color=0;
dot[x].nxt-x=NO-DOT;
dot[x].pri-x=NO-DOT;
if((x+EYE-SPACE)<WIDTH) dot[x].nxt-x=x+EYE-SPACE;
if((signed)x-EYE-SPACE)>=0)dot[x].pri-x=x-EYE-SPACE;
}
for(layer=1;layer<NUM-COLOR;layer++)
for(x=0;x<WIDTH;x++) {
left-x=x-EYE-SPACE/2+(layer/2);
right-x=x+EYE-SPACE/2-((layer+1)/2);
if((h[x]==layer)&&(left-x>=0)&&(rignt-x<WIDTH)) {
if (dot[left-x].nxt-x!=NO-DOT) dot[dot[left-x].nxt-x].pri-x=NO
-DOT;
dot[left-x].nxt-x=right-x;
if (dot[right-x].pri-x!=NO-DOT) dot[dot[right-x].nxt-x].nxt-x=
NO-DOT;
dot[right-x].pri-x=left-x;
}
}
for(x=0;x<WIDTH;x++) {
if(dot[x].pri-x==NO-DOT) {
dot[x].color=org-color[x];
tmp-x=x;
while( dot[tmp-x].nxt-x!=NO-DOT) {
tmp-x=dot[tmp-x].nxt-x;
dot[tmp-x].color=org-color[x];
}
}
}
fseek(fResult,cur-offset,SEEK-SET);
for(i-byte=0;i-byte<BYTE-PER-LINE;i-byte++) {
tmp-bytel=0;
for(i-pixel=0;i-pixel<PIXEL-PER-BYTE;i-pixel++) {
x=i-byte*PIXEL-PER-BYTE+i-pixel;
if(x<WIDTH)
tmp-byte1=(unsigned char) (tmp-byte1<<(BITS-PER-PIXEL*i
-pixel)) +dot[x].color;
}
fwirte(&tmp-byte1,1,1,fResult);
}
}
fclose(fLayer);fclose(fOrigin);fclose(fResult);
}

参考文献
孙志辉、王萃寒、王茜.实用Windows 3.1详解.北京:电子工业出版社,1994.

转贴于 中国文秘网 http://www.zgwmw.com
《用计算机程序制作三维立体画》来源于中国文秘网,中国最专业的文秘网站,欢迎阅读用计算机程序制作三维立体画。
论文录入:中国文秘网    责任编辑:中国文秘网 
  • 上一篇论文:

  • 下一篇论文:
  • 发表评论】【加入收藏】【告诉好友】【打印此文】【关闭窗口
    最新热点 最新推荐 相关论文
    计算机不能思维的初步逻辑证…
    从小学计算机教学实践谈学生…
    浅谈中年级计算机教学
    脉冲功率装置能源计算机控制…
    计算机网络系统在制碱工业中…
    浅谈计算机多媒体在教学中的…
    银行计算机网络风险防范与对…
    PowerBuilder的分布式计算技…
    浅谈计算机网络环境下的注册…
    浅谈《高中地理》计算机辅助…
      网友评论:(只显示最新10条。评论内容只代表网友观点,与本站立场无关!)
    sitemap:1 2 3 4 5 6 7 8 9 10 [11] 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
    28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
    55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
    82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
    109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
    136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162
    163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
    190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216
    217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243
    244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270
    271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297
    298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324
    325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351
    352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378
    379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405
    406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432
    433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459
    460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480