• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 132  文章 - 51  trackbacks - 0
            <2011年9月>
            28293031123
            45678910
            11121314151617
            18192021222324
            2526272829301
            2345678

            常用鏈接

            留言簿(7)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            cocos2d-x

            OGRE

            OPenGL

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

                  在讀取BMP圖像的時候,主要處理了24位和32位圖像,在處理24位BMP圖像的時候,BMP圖像的數據已經被修改以便4字節自動對齊,
             對于24位BMP圖像而言,最需要注意的一點是,規定了每條行掃描線的數據大小必須是4的整數倍,如果不是4的整數倍,那么需要在行末端進行補0,否則數據讀取將出現偏移,直接導致的結果:會加載出一張傾斜的圖像。這個補0的操作叫做數據寬度對齊。
                  很抽象?舉個例子,對于400×400的24位BMP圖像而言,行掃描是400px,是4的整數倍,那么無需進行補0操作。如果是30×38(本篇每張麻將牌素材的尺寸),那么行掃描顯然不是4的整數倍。對于24位BMP而言,一個像素由RGB三字節組成,那么一個行掃描的total字節是30 × 3 = 90字節,進行數據寬度對齊之后,行掃描的實際total字節是30 × 3 + 2 = 92字節。如此,才能被4整除。換句話說,每一行多出2個無用的字節。這在編程中,需要進行處理丟棄,否則圖像再次變形
                

                        int channel= 0;
                        
            if ( texture->bpp == 24 ){
                            channel 
            = 3;
                            texture
            ->imageType = GL_RGB;
                        }
            else{
                            channel 
            = 4;
                            texture
            ->imageType = GL_RGBA;
                        }

                        
            int biWidth = 0;                                                //每行補齊字節數,如果是24位的話需要補齊成4字節的倍數
                        if ( (texture->width*channel) % 4 != 0 )
                            biWidth 
            = 4 - ( (texture->width*channel) % 4 );
                        
            我們求出每行自動填充的字節數,先讀取數據,然后跳過自動添加的字節數,再接著讀取下一行數據
                        forint i = 0; i < height; i++ ){
                            GLubyte
            * pdest = texture->imageData + i * width * channel;
                            fread( pdest, 
            1, width * channel,file );

                            
            forint j = 0; j < width* channel; j+=channel ){
                                pdest[ j ] 
            ^= pdest[j+2^= pdest[j] ^= pdest[j+2];                //將BGR轉為RGB,BMP數據從左下角到右上角的方式存儲
                            }

                            
                            fseek( file, biWidth, SEEK_CUR );                                    
            //丟棄補足字節,開始填充下一行    
                        }

            讀取數據完畢,可以打印檢驗一下,用WINHEX打開看下,NoPropblem
            因為我讀取的圖片不全是512*512,256*256,.......,所以我用OPENGL生成紋理的時候用了個gluBuild2DMipmaps
            BOOL bRes = gluBuild2DMipmaps( GL_TEXTURE_2D,  texture->bpp/8, texture->width,texture->height, texture->imageType, GL_UNSIGNED_BYTE, texture->imageData );
            畫出來后又成斜著的圖像了,數據也沒問題,只能說是生成MipMap 的時候讀取數據跨度有問題,但是我傳遞的是32位圖像的時候,生成的MIPMAP沒任何問題,即使圖像大小不能被2整除,即非512*512,256*256,.......
            所以我做了個比較惡心的做法
                if ( bMipmap ){
                    
            if ( texture->imageType == GL_RGB ){
                        GLubyte
            * data = new GLubyte[texture->width*texture->height*4 ];
                        
            int counter = 0;
                        
            forint i = 0; i < texture->width*texture->height*3; i+=3 ){
                            data[i
            +counter] = texture->imageData[i];
                            data[i
            +1+counter] = texture->imageData[i+1];
                            data[i
            +2+counter] = texture->imageData[i+2];
                            data[i
            +3+counter] = 1.f;
                            counter
            ++;
                        }

                        BOOL bRes 
            = gluBuild2DMipmaps( GL_TEXTURE_2D,  4, texture->width,texture->height, GL_RGBA, GL_UNSIGNED_BYTE, data );
                        delete[] data;
                    }
            else{
                        BOOL bRes 
            = gluBuild2DMipmaps( GL_TEXTURE_2D,  texture->bpp/8, texture->width,texture->height, texture->imageType, GL_UNSIGNED_BYTE, texture->imageData );
                    }

                    
                    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR_MIPMAP_LINEAR );
                    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);    

            將24位的圖像生成紋理的時候,我把數據轉換為32位的就沒問題了,一直沒更好的辦法,請大家不吝指導

            題外文章:
            BMP文件的結構,從大的分類來看,可以分成2部分:頭部信息區和圖像數據區。其中頭部信息區中,保存了圖像的各種屬性,如文件格式,圖片寬度,調色板等等。過了頭部信息區之后,就是用來呈現圖像的真正的數據區了。BMP文件屬于像素文件,也就是說,數據區中的數據,其實就是記錄了圖像中每一個像素的顏色,以32位BMP圖片來說,每一個像素由ARGB四個字節保存其顏色,其中A是透明度。那么,一副400×400的32位BMP圖像,就由160000個像素組成,則,數據區大小=160000×4byte = 640000byte = 640kb,所以也不難怪,BMP圖像文件給人的印象就是文件非常大。

                   那么將BMP文件中的像素逐個讀入數據緩沖再顯示,是否就能正常顯示了呢?顯然不是的。前面說了,在文件開頭有一塊頭部信息區,如果把這些數據也一起讀進緩沖,那么圖像是無法最終顯示的。因此,通常載入BMP圖像的首要任務是要知道這塊頭部信息區的大小,也就是數據區的偏移量offset,再利用fseek尋址到這個位置,再進行逐個像素的解析。而頭部信息區的大小值貌似在頭部信息區的某個字節中有的,只需要讀取該字節,便可知道。對于24位和32位的BMP圖像而言,這個offset值為0x36,也就是頭部信息區的大小是54個字節。

                   如果你要把加載圖像的函數做的通用一些的話,那么圖片的寬度和高度也是需要獲取的,對于24位BMP圖像,這兩個值可以分別在0x12和0x16進行尋址得到,注意fread需要讀雙字,即4字節,否則會出錯。

                   對于24位BMP圖像而言,最需要注意的一點是,規定了每條行掃描線的數據大小必須是4的整數倍,如果不是4的整數倍,那么需要在行末端進行補0,否則數據讀取將出現偏移,直接導致的結果:會加載出一張傾斜的圖像。這個補0的操作叫做數據寬度對齊。

                   很抽象?舉個例子,對于400×400的24位BMP圖像而言,行掃描是400px,是4的整數倍,那么無需進行補0操作。如果是30×38(本篇每張麻將牌素材的尺寸),那么行掃描顯然不是4的整數倍。對于24位BMP而言,一個像素由RGB三字節組成,那么一個行掃描的total字節是30 × 3 = 90字節,進行數據寬度對齊之后,行掃描的實際total字節是30 × 3 + 2 = 92字節。如此,才能被4整除。換句話說,每一行多出2個無用的字節。這在編程中,需要進行處理丟棄,否則圖像再次變形。

                   最后一個需要注意的小細節是,BMP圖像的原點坐標,都是以左下角為基準,向右、向上增加。所以,在編程時,需要對y軸數據做一些小變換。否則將會得到一幅顛倒的圖像,此外,BMP的三原色順序是BGR,注意編程中的處理。


            原文地址:BMP文件的結構,從大的分類來看,可以分成2部分:頭部信息區和圖像數據區。其中頭部信息區中,保存了圖像的各種屬性,如文件格式,圖片寬度,調色板等等。過了頭部信息區之后,就是用來呈現圖像的真正的數據區了。BMP文件屬于像素文件,也就是說,數據區中的數據,其實就是記錄了圖像中每一個像素的顏色,以32位BMP圖片來說,每一個像素由ARGB四個字節保存其顏色,其中A是透明度。那么,一副400×400的32位BMP圖像,就由160000個像素組成,則,數據區大小=160000×4byte = 640000byte = 640kb,所以也不難怪,BMP圖像文件給人的印象就是文件非常大。

                   那么將BMP文件中的像素逐個讀入數據緩沖再顯示,是否就能正常顯示了呢?顯然不是的。前面說了,在文件開頭有一塊頭部信息區,如果把這些數據也一起讀進緩沖,那么圖像是無法最終顯示的。因此,通常載入BMP圖像的首要任務是要知道這塊頭部信息區的大小,也就是數據區的偏移量offset,再利用fseek尋址到這個位置,再進行逐個像素的解析。而頭部信息區的大小值貌似在頭部信息區的某個字節中有的,只需要讀取該字節,便可知道。對于24位和32位的BMP圖像而言,這個offset值為0x36,也就是頭部信息區的大小是54個字節。

                   如果你要把加載圖像的函數做的通用一些的話,那么圖片的寬度和高度也是需要獲取的,對于24位BMP圖像,這兩個值可以分別在0x12和0x16進行尋址得到,注意fread需要讀雙字,即4字節,否則會出錯。

                   對于24位BMP圖像而言,最需要注意的一點是,規定了每條行掃描線的數據大小必須是4的整數倍,如果不是4的整數倍,那么需要在行末端進行補0,否則數據讀取將出現偏移,直接導致的結果:會加載出一張傾斜的圖像。這個補0的操作叫做數據寬度對齊。

                   很抽象?舉個例子,對于400×400的24位BMP圖像而言,行掃描是400px,是4的整數倍,那么無需進行補0操作。如果是30×38(本篇每張麻將牌素材的尺寸),那么行掃描顯然不是4的整數倍。對于24位BMP而言,一個像素由RGB三字節組成,那么一個行掃描的total字節是30 × 3 = 90字節,進行數據寬度對齊之后,行掃描的實際total字節是30 × 3 + 2 = 92字節。如此,才能被4整除。換句話說,每一行多出2個無用的字節。這在編程中,需要進行處理丟棄,否則圖像再次變形。

                   最后一個需要注意的小細節是,BMP圖像的原點坐標,都是以左下角為基準,向右、向上增加。所以,在編程時,需要對y軸數據做一些小變換。否則將會得到一幅顛倒的圖像,此外,BMP的三原色順序是BGR,注意編程中的處理。


            原文地址:http://naozifangde.blog.163.com/blog/static/1280042642009101614938531/
            posted on 2010-05-29 11:22 風輕云淡 閱讀(2647) 評論(0)  編輯 收藏 引用 所屬分類: 圖像讀取
            久久强奷乱码老熟女网站| 国产精品禁18久久久夂久| WWW婷婷AV久久久影片| 亚洲天堂久久久| 精品欧美一区二区三区久久久| 国产91色综合久久免费| 国内精品伊人久久久久777| 久久免费视频一区| 日韩欧美亚洲综合久久影院Ds | 久久久久亚洲av无码专区喷水| 久久久噜噜噜久久中文福利| 精品国产青草久久久久福利| 国产69精品久久久久APP下载| 久久笫一福利免费导航| 青青久久精品国产免费看| 欧美伊人久久大香线蕉综合69 | 热re99久久精品国99热| 久久久噜噜噜久久| 久久久久无码精品| 性高湖久久久久久久久AAAAA| 久久国产欧美日韩精品免费| 欧美亚洲国产精品久久高清| 亚洲va久久久噜噜噜久久| 久久久久亚洲AV无码麻豆| 99精品久久精品一区二区| 大美女久久久久久j久久| 人妻无码精品久久亚瑟影视| 亚洲级αV无码毛片久久精品| 欧美牲交A欧牲交aⅴ久久| 久久精品人人槡人妻人人玩AV| 久久久久亚洲av无码专区导航| 色综合久久最新中文字幕| 久久乐国产综合亚洲精品| 久久99精品久久久久久久不卡| 精品多毛少妇人妻AV免费久久| 四虎国产精品成人免费久久| 国产精品一久久香蕉国产线看 | 久久人人超碰精品CAOPOREN| 久久午夜综合久久| 午夜天堂av天堂久久久| 超级碰久久免费公开视频|