當前位置:首頁 > 智能硬件 > 智能硬件
[導讀]摘 要 該文介紹了三維立體畫的原理和制作 方法 ,并給出了用C語言編寫的源程序。借助于Pbrush.exe,讀者可以自己設(shè)計和欣賞各式各樣的三維立體畫。目前 ,市面上正在流行各式各樣的立體畫,其特點是從外表來看與一般的圖

摘 要 該文介紹了三維立體畫的原理和制作 方法 ,并給出了用C語言編寫的源程序。借助于Pbrush.exe,讀者可以自己設(shè)計和欣賞各式各樣的三維立體畫。

目前 ,市面上正在流行各式各樣的立體畫,其特點是從外表來看與一般的圖案很相似,但是雙眼緊盯著注視片刻后,一恍惚之間眼前便出現(xiàn)了畫中畫——立體像。筆者第一次看到這種畫便被發(fā)明者的創(chuàng)意所傾倒。利用眾所皆知的雙眼視差原理,竟能在一張平面紙上制造出如此奇幻。但是立體畫本身除了其發(fā)明者的靈感和畫面創(chuàng)作者的別出心裁之外,其原理上并無神秘之處。用 計算 機程序來實現(xiàn)它,可說是易如反掌。筆者用一個晚上時間,便在微機上用BASIC語言實現(xiàn)了簡單形體——平面圓餅的立體畫。當然,要使該程序具有完善的功能,提高其制作速度,還是應該用編譯語言(如C語言)來編寫。本文中給出的源程序借助于Windows中的.BMP圖形文件,可使大家自己制作任意形態(tài)的立體畫。

一、立體畫的原理

看過立體電影的人都知道,當人的雙眼分別接收不同視角拍攝的圖像時便會產(chǎn)生立體感。這是由于人眼長期觀察的習慣造成的。和立體電影原理相同的立體攝影風景照片也很早就已出現(xiàn)。圖1中給出了這種立體照片的示意圖。左、右照片分別是人的雙眼角度上觀察一棱錐體時左右眼看到的圖像(圖2)。左眼看到的是棱錐的頂端向右錯動了一些的圖像,右眼的看到則是棱錐的頂端向左錯動了一些的圖像。如果用一張硬卡片隔開兩張照片(如圖3),@@09A04000.GIF;圖1@@

@@09A04001.GIF;圖2@@

@@09A04002.GIF;圖3雙眼分別看兩張畫,會看到一個立體的棱錐體。這種立體照片的觀察方法在測繪學中也早已采用。

但是,目前的三維立體畫在形式上與這些很不相同。它是怎樣在同一張畫面上呈現(xiàn)立體的呢?首先, 分析 一下人們是怎樣從這些立體畫中看出“立體形體”的。從前面所說的可以知道,人眼要得到立體感,雙眼必須有視差,即雙眼看到的圖像應該有差異。人們在看立體畫時,都有“恍惚”一下的過程。在這過程中,雙眼的視中心發(fā)生了錯動(如圖4)。這樣@@09A04003.GIF;圖4左眼看到的是畫面的“偏左像”,右眼看到的是畫面的“偏右像”。@@只要“偏左像”和“偏右像”的 內(nèi)容 相當于圖1的左、右照片,雙眼就會感到立體形體。那么,能否把圖1的左、右照片分別當做“偏左像”和“偏右像”,簡單重疊來得到立體畫呢?

顯然不行。能夠合成立體畫的“偏左像”和“偏右像”是要滿足一定條件的。

如果圖5中表現(xiàn)的棱錐體的表面上有圖案的話,

@@09A04004.GIF;圖5像素a和像素a#39;應該具有相同的顏色,因為它們是從不同視角觀察的@@同一個實體點。像素b和像素b#39;、像素c和像素c#39;的情況與此相同。把兩幅畫分別當作“偏左圖”和“偏右圖”,部分重疊成為同一畫面時,在新的畫面上這種關(guān)系仍應該表現(xiàn)為a=a#39;,b=b#39;,c=c#39;(如圖6)。但這時應該注意到,在這張合成

@@09A04005.GIF;圖6畫面上,點a#39;既是“偏右圖”上的點a#39;,又是“偏左圖”上的點b。而@@一張畫面上相同坐標點的像素只可能是一種顏色,因此,產(chǎn)生了新的像素關(guān)系a#39;=b。另外,點a既是“偏左圖”上的點a,又是“偏右圖”上的點c#39;,所以,a=c#39;。以此類推,點b#39;和點c也有類似的情況。因此出現(xiàn)了新的關(guān)系表示式,...#39;=c=c#39;=a=a#39;=b=b#39;=...。這就構(gòu)成了立體畫面上像素必須要滿足的條件:“等顏色像素鏈”。立體畫上的所有點都從屬于某一條“等顏色像素鏈”。這就是所有立體畫圖案都呈現(xiàn)出某種程度上的水平周期性的原因。

因此,對于任意立體形狀,只要構(gòu)造出相應的這種“等像素鏈”,并按其 規(guī)律 充填圖案即可得到立體畫。但是正如前面所述,由于這種“等像素鏈”條件的約束,人們雖然可以隨意構(gòu)造出各種形體的立體畫,但其立體形體的表面圖案是不能完全隨人意愿的。

二、制作立體畫的計算機程序

由于人的雙眼的水平性,以上的“等像素鏈”只按水平方向分布,與垂直方向無關(guān)。因此,在程序中,各個像素行的處理過程是相互獨立的。制作立體畫的程序主結(jié)構(gòu)圖如圖7。

@@09A04006.GIF;圖7 制作立體畫的程序主結(jié)構(gòu)圖在以上結(jié)構(gòu)圖中,關(guān)鍵是如何建立“等@@像素鏈”。具體的處理如下。對于立體形體上的每一個點,首先求出該點在“偏左圖”和“偏右圖”上的坐標。以圖1中的棱錐頂點為例,實際上其X坐標是在中心點,但由于雙眼的位置并不在其正上方,頂點在“偏左圖”上向右位移,在“偏右圖”上向左位移,而且其位移值的大小顯然與其高度有關(guān),即該點坐標越高位移值就越大。a,b,c等點也都有這些位移。在求出一個點在“偏左圖”和“偏右圖”上的坐標后,再算出在合成圖(如圖6)上的對應坐標

,以建立“等像素”關(guān)系,如a=a#39;。當立體形體的一個水平剖面上的全部點經(jīng)過以上處理后,合成圖的各條“等像素鏈”關(guān)系也就 自然 形成了。

另外,由于有可能出現(xiàn)高點遮蓋低點的情況,“等像素鏈”的構(gòu)造應該從低點到高點逐層進行,高點的“等像素”關(guān)系將替代低點的“等像素”關(guān)系。這也是程序主結(jié)構(gòu)圖中“首先,對于沒有任何形體存在的背景平面構(gòu)造‘等像素鏈’”的原因。

下面給出了根據(jù)以上結(jié)構(gòu)圖用C語言編寫的源程序。程序中,每一個坐標點對應一個結(jié)構(gòu)型數(shù)據(jù),它包含“前像素”、“后像素”兩個指針。“前像素”指針指向該坐標點作為“偏右圖”上的一點,在“偏左圖”所對應的點的坐標。“后像素”指針指向該坐標點作為“偏左圖”上的一點,在“偏右圖”所對應的點的坐標。程序中,“立體形體水平剖面的高低坐標數(shù)據(jù)”、“原始圖案素材”和輸出的“立體畫”的文件格式都是采用了Windows3.1 的Pbrush產(chǎn)生的BMP圖形文件格式。圖幅大小要求都是640×400,用16種顏色方式。其中,立體形體上各點的高低坐標用圖形文件中的顏色值表示,因此該圖形文件的圖形與帶顏色的等高線圖安全相同。通常情況下,在16色的BMP文件中顏色值從小到大的順序為:黑色、暗紅色、暗綠色、暗黃色、暗藍色、暗紫色、暗青色、暗灰色、灰色、明紅色、明綠色、明黃色、明藍色、明紫色、明青色、白色。本程序采用最簡單的“圖案充填”方案,即各條“鏈”上的像素點皆采用該“鏈”上的第一個像素的顏色。程序中的常數(shù)EYE-SPACE表示“偏左圖”和“偏右圖”之間的偏差,BO-DOT是表明“鏈”的首或尾的指針標志。

該程序寄生在Windows 3.1中的Pbrush軟件上。借助于它來構(gòu)筑立體形體(即立體形體水平剖面高低坐標數(shù)據(jù)文件圖8),設(shè)計原始圖案(圖9)。程序運行后,逐行輸入并處理以上兩個文件中的圖形,然后輸出立體畫結(jié)果文件(圖10)。最后,用Pbrush來觀賞立體畫result.bmp。當然,要設(shè)計出令人賞心悅目的立體畫,必須在立體形體和圖案素材的選擇和搭配上做到天衣無縫,獨具匠心。

@@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#39;BM#39;*/

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==#39;B#39; && BITMAPFILEHEADER.bfType2==#39;M#39;

&& 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! ",fn-layer);

getch();

exit(1);

}

}

else{

printf("File %s does not exist! ",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==#39;B#39; && BITMAPFILEHEADER.bfType2==#39;M#39;

&& 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! ",fn-org);

getch();

exit(3);

}

}

else {

printf("File %s does not exist! ",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! ",fn-result);

getch();

exit(5);

}

for(line=0;line<NUM-LINE;line++){

printf("line=%d ",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詳解.北京: 電子 工業(yè) 出版社,1994.

更多計算機與外設(shè)信息請關(guān)注21ic計算機與外設(shè)頻道

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉