• <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>

            無(wú)我

            讓內(nèi)心永遠(yuǎn)燃燒著偉大的光明的精神之火!
            靈活的思考,嚴(yán)謹(jǐn)?shù)膶?shí)現(xiàn)
            豪邁的氣魄、頑強(qiáng)的意志和周全的思考

            eSNACC對(duì)INTEGER的編碼和解碼

            本文剖析asn-int.h/c,從源代碼來(lái)學(xué)習(xí)eSNACC對(duì)INTEGER的編碼和解碼。

            eSNACC中的整形與上一篇所講的布爾型一樣,也很簡(jiǎn)單,不過(guò)代碼中還是有一點(diǎn)有趣的地方。

             

            eSNACC定義的整形AsnInt是用4字節(jié)的整數(shù)表示的。如下代碼

            #if SIZEOF_INT == 4
            #  define I        
            int
            #else
            #  
            if SIZEOF_LONG == 4
            #    define I        
            long
            #  
            else
            #    
            if SIZEOF_SHORT == 4
            #      define I        
            short
            #    endif
            #  endif
            # endif

            #ifdef I
              typedef I        AsnInt;
              typedef unsigned I    UAsnInt;
            #else
              
            #error "can't find integer type which is 4 bytes in size"
            #endif
            #undef I

            其中用了一段預(yù)處理指令來(lái)取得當(dāng)前機(jī)器或編譯器的4字節(jié)類型(int,long or short)。同時(shí)定義了無(wú)符號(hào)的整形UAsnInt。

             

            我們著重研究一下.c文件中的編碼和解碼。

            /*
             * encodes signed long integer's contents
             
            */

            AsnLen
            BEncAsnIntContent PARAMS ((b, data),
                GenBuf 
            *b _AND_
                AsnInt 
            *data)
            {
                
            int             len;
                
            int             i;
                unsigned 
            long  mask;
                
            long dataCpy;

            #define INT_MASK (0x7f80 << ((sizeof(AsnInt) - 2) * 8))

                dataCpy 
            = *data;

                
            /*
                 * calculate encoded length of the integer (content)
                 
            */

                mask 
            = INT_MASK;
                
            if (dataCpy < 0)
                     
            for (len = sizeof (AsnInt); len > 1--len)
                     
            {
                         
            if ((dataCpy & mask) == mask)
                            mask 
            >>= 8;
                        
            else
                            
            break;
                    }

                
            else
                    
            for (len = sizeof (AsnInt); len > 1--len)
                    
            {
                        
            if ((dataCpy & mask) == 0)
                            mask 
            >>= 8;
                        
            else
                            
            break;
                    }


                
            /*
                 * write the BER integer
                 
            */

                
            for (i = 0; i < len; i++)
                
            {
                    BufPutByteRvs (b, (unsigned 
            char)dataCpy);
                    dataCpy 
            >>= 8;
                }


                
            return len;

            }
              /* BEncAsnIntContent */


            /*
             * Decodes content of BER a INTEGER value.  The given tag is ignored.
             
            */

            void
            BDecAsnIntContent PARAMS ((b, tagId, len, result, bytesDecoded, env),
                GenBuf 
            *b _AND_
                AsnTag     tagId _AND_
                AsnLen     len _AND_
                AsnInt    
            *result _AND_
                AsnLen 
            *bytesDecoded _AND_
                jmp_buf env)
            {
                
            int   i;
                
            long retVal;
                unsigned 
            long byte;


                
            if (len > sizeof (AsnInt))
                
            {
                    Asn1Error (
            "BDecAsnIntContent: ERROR - integer to big to decode.\n");
                    longjmp (env, 
            -7);
                }


                
            /*
                 * look at integer value
                 
            */

                
            byte = (unsigned long ) BufGetByte (b);

                
            if (byte & 0x80)   /* top bit of first byte is sign bit */
                    retVal 
            = (-1 << 8| byte;
                
            else
                    retVal 
            = byte;

                
            /*
                 * write from buffer into long int
                 
            */

                
            for (i = 1; i < (int)len; i++)
                    retVal 
            = (retVal << 8| (unsigned long)(BufGetByte (b));

                
            if (BufReadError (b))
                
            {
                    Asn1Error (
            "BDecAsnIntContent: ERROR - decoded past end of data \n");
                    longjmp (env, 
            -8);
                }

                (
            *bytesDecoded) += len;

                
            *result = retVal;
                tagId
            =tagId;  /* referenced to avoid compiler warning. */

            }
              /* BDecAsnIntContent */

            這些代碼最神奇的地方就是專門設(shè)計(jì)了這樣一個(gè)標(biāo)志

            #define INT_MASK (0x7f80 << ((sizeof(AsnInt) - 2) * 8))

            可以說(shuō),一切精華都在這里面。

            目的:在實(shí)際中,整形是一個(gè)4字節(jié)數(shù)來(lái)表示的,但是在編碼傳輸時(shí),我們希望盡最大可能來(lái)壓縮這個(gè)數(shù),也就是取得這個(gè)數(shù)的最多有效字節(jié)。

            然后就可以分情況討論了:

            1.若值data<0:那么其最高位一定為1,那么我們就是要最大限度的壓縮從最高位開始的bit位為1的字節(jié)。只要這些bit位為1,我們就可以壓縮掉了。這也就是以下代碼的目的:

            for (len = sizeof (AsnInt); len > 1--len)
                     
            {
                         
            if ((dataCpy & mask) == mask)
                            mask 
            >>= 8;
                        
            else
                            
            break;
                    }

             

            2.若值data>0:那么其最高位一定為0,那么我們就是要最大限度的壓縮從最高位開始的bit位為0的字節(jié)。只要這些bit位為0,我們就可以壓縮掉了。這也就是以下代碼的目的:
            for (len = sizeof (AsnInt); len > 1--len)
                    
            {
                        
            if ((dataCpy & mask) == 0)
                            mask 
            >>= 8;
                        
            else
                            
            break;
                    }

             

            然后在解碼時(shí),就要根據(jù)編碼的這些邏輯來(lái)判斷:如果字節(jié)的最高位為1,則說(shuō)明這個(gè)數(shù)是個(gè)負(fù)數(shù),所以就先將這個(gè)字節(jié)的前面填充1擴(kuò)展。否則,就是正數(shù),不需要變:

            if (byte & 0x80)   /* top bit of first byte is sign bit */
                    retVal 
            = (-1 << 8| byte;
                
            else
                    retVal 
            = byte;

            然后在根據(jù)有效字節(jié)長(zhǎng)度來(lái)擴(kuò)展還原為本來(lái)的值。

            for (i = 1; i < (int)len; i++)
                    retVal 
            = (retVal << 8| (unsigned long)(BufGetByte (b));

             

             

            依據(jù)上面的思想,本文件中的其他函數(shù)也就不難理解了。只是在編碼無(wú)符號(hào)數(shù)時(shí),為了避免混淆,多引入了一個(gè)字節(jié)而已。

             

            不過(guò),我感到最值得我們思考和學(xué)習(xí)的是這個(gè)mask是如何設(shè)計(jì)出來(lái)的?我們?cè)撛趺此伎疾拍茉O(shè)計(jì)出這么神奇的數(shù)呢?這個(gè)還請(qǐng)有經(jīng)驗(yàn)的朋友總結(jié)和賜教。

            posted on 2012-04-20 17:26 Tim 閱讀(1704) 評(píng)論(4)  編輯 收藏 引用 所屬分類: eSNACC學(xué)習(xí)

            評(píng)論

            # re: eSNACC對(duì)INTEGER的編碼和解碼 2012-04-20 18:13 Silk top virgin glueless lace wigs

            經(jīng)典,eSNACC的整形還真深?yuàn)W  回復(fù)  更多評(píng)論   

            # re: eSNACC對(duì)INTEGER的編碼和解碼 2012-04-21 08:44 周星星

            0x7f80 就是一個(gè)連續(xù)8bit為1的數(shù),也就是 0xFF 右移了1bit
            之所以不用0xFF,是因?yàn)槭孜灰呀?jīng)有 if(xxx<0) else 判斷了  回復(fù)  更多評(píng)論   

            # re: eSNACC對(duì)INTEGER的編碼和解碼 2012-04-21 16:42 Tim

            設(shè)計(jì)成0xFF是非常有遠(yuǎn)見的方案,不過(guò)千萬(wàn)不能遺忘或者后面這個(gè)80了。因?yàn)槲覀儽仨氁WC要舍棄的字節(jié)后面字節(jié)的第一位與標(biāo)記位完全相同!
            另外,我想了想,雖然以前在外部判斷了正負(fù)了,但是好像mask=0xFF80也是可以的,覺(jué)得這樣是不是更容易理解呢?@周星星
              回復(fù)  更多評(píng)論   

            <2009年12月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            導(dǎo)航

            統(tǒng)計(jì)

            公告

            本博客原創(chuàng)文章,歡迎轉(zhuǎn)載和交流。不過(guò)請(qǐng)注明以下信息:
            作者:TimWu
            郵箱:timfly@yeah.net
            來(lái)源:www.shnenglu.com/Tim
            感謝您對(duì)我的支持!

            留言簿(9)

            隨筆分類(173)

            IT

            Life

            搜索

            積分與排名

            最新隨筆

            最新評(píng)論

            閱讀排行榜

            日韩十八禁一区二区久久| 久久免费精品视频| 久久丝袜精品中文字幕| 色综合久久88色综合天天| 久久99久久99精品免视看动漫| 亚洲中文字幕无码久久2017| 久久99久久99精品免视看动漫| 久久亚洲精品无码VA大香大香 | 中文字幕无码久久精品青草| 久久精品一区二区三区中文字幕| 国产高潮久久免费观看| 久久国产成人午夜aⅴ影院 | 久久亚洲国产中v天仙www| 久久精品国产精品青草| 国产精品免费久久久久影院 | 午夜精品久久久久久久无码| 性欧美大战久久久久久久 | 久久亚洲AV成人无码国产| AV无码久久久久不卡网站下载| 精品久久久久久亚洲精品 | 久久最新精品国产| 久久AⅤ人妻少妇嫩草影院| 综合久久给合久久狠狠狠97色 | 国产91久久综合| 热99RE久久精品这里都是精品免费| 97久久国产综合精品女不卡| 国产精品久久久久9999高清| 久久精品无码一区二区日韩AV| 亚洲va久久久久| 岛国搬运www久久| 成人午夜精品无码区久久| a级毛片无码兔费真人久久| 狠狠色婷婷久久综合频道日韩| 婷婷综合久久中文字幕| 婷婷久久五月天| 国产精品免费久久久久久久久| 99久久精品国产一区二区| 91亚洲国产成人久久精品| 伊人久久久AV老熟妇色| 久久国产精品国语对白| 亚洲精品乱码久久久久久自慰|