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

            那誰的技術博客

            感興趣領域:高性能服務器編程,存儲,算法,Linux內核
            隨筆 - 210, 文章 - 0, 評論 - 1183, 引用 - 0
            數據加載中……

            如何使用位操作得到大于N且為2的次方的最小的數

            RT,比如輸入10返回16, 輸入24返回32,等等.


            注意,是使用位操作且沒有循環,也不用表驅動等等.因為這個操作要經常進行,要保證高效,所以不能使用循環(別跟我說用遞歸,熟悉算法和計算機本質的人都知道遞歸和循環本質是一樣的:);同時,因為不知道需要計算的數據到底有多大,采用表驅動的辦法也不可行.

            我在網上發帖,最終得到了一個很BT的答案:
            int fun(int v)
            {
                
            float f = (float)(v - 1);  
                
            return 1 << ((*(unsigned int*)(&f) >> 23- 126);
            }

            但是我不知道這個算法的原理是什么,貌似采用了浮點數格式的一些特性,知道的同學請給我一個詳盡的解釋,在這里先感謝了.

            posted on 2008-06-21 15:36 那誰 閱讀(4784) 評論(10)  編輯 收藏 引用 所屬分類: C\C++算法與數據結構

            評論

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            很邪惡啊,我還是喜歡簡單一些的。。。

            int fun(int v)
            {
            int res = 1;
            while (res < v)
            res <<= 1;

            return res;
            }
            2008-06-21 16:36 | 大日如來

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            其實也不復雜,只要知道float在內存里是怎么存儲的就簡單的很了
            IEEE:
            float:
            Sign Exponent Mantissa
            1bit 8bits 23bits

            實際上float的Mantissa是24位,有一個隱含的最高位,且該位一直為‘1’

            Mantissa表示1~2之間的數
            Expoent=0x7f表示指數0, Expoent>0x7f為正指數,Expoent<0x7f則為負指數
            2008-06-21 16:41 | 大日如來

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            和BIG-ENDIAN or LITTLE_ENDIAN有關,推薦少用。
            2008-06-21 16:51 | 大日如來

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            @大日如來
            我不覺得和大小端有關系,浮點寄存器和整數寄存器在內存存取上是一致的。
            2008-06-22 17:50 | Louix

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            剛才做實驗驗證了一下,這個方法是有Bug的,使用float類型會造成轉換時的進位,使用double就可以了,求數字二進表示長度的代碼:
            double dTarget = ( double )target;
            return ( ( *( ( ( unsigned int * ) &dTarget ) + 1 ) ) >> 20 ) - 1023;
            這樣的代碼就和大小端相關了,而且沒有其他版效率高。
            2008-06-22 18:45 | Louix

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            答案是很BT的,我是想不到
            2008-06-22 23:00 | 影視劇

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            這個算法只是玩一玩了,有助于了解IEEE標準浮點數的構造。
            實際應用中整形和浮點的相互轉換很花時間,還不如一步步位移來做。
            2008-06-25 09:20 | RedNax

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            解決問題的關鍵在于如何找到二進制表示時第一個‘1’出現的位置。‘1’出現在第n位,則將1左移n位即可。
            v - 1是不必要的,意圖可能是想去掉大日如來所說的“隱含的最高位1”,直接轉就可以了,要了反而算出的結果不對,不是float類型轉換進位的問題。
            f右移23位為的是得到存儲在Exponent里的(n-1)+127=n+126,127是常量。再減去126求得n。
            利用浮點數存儲格式來找到這個n確實是我沒想到的,想出這個解法的人思維很發散。
            2008-07-21 15:51 | wolfssss

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            @wolfssss
            float f = (float)(v - 1);
            v減不減一的區別在于

            float f = (float)(v - 1);
            求的是大于等于N且為2的次方的最小的數

            float f = (float)(v);
            求的是大于N且為2的次方的最小的數
            2011-10-16 12:16 | 楚天清秋

            # re: 如何使用位操作得到大于N且為2的次方的最小的數  回復  更多評論   

            可以下ieee 754 就明白了
            2013-02-22 17:07 | mankeyl
            久久99精品久久久大学生| 久久大香萑太香蕉av| 亚洲色大成网站www久久九| 久久精品中文字幕一区| 亚洲AV无码久久寂寞少妇| 亚洲va国产va天堂va久久| 久久精品国产亚洲一区二区| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 漂亮人妻被黑人久久精品| 久久九九精品99国产精品| 国产叼嘿久久精品久久| 久久婷婷午色综合夜啪| 97热久久免费频精品99| 久久精品国产99久久久香蕉| 亚洲AV日韩精品久久久久久| 国产精品久久久久aaaa| 亚洲欧洲久久久精品| 久久综合九色综合精品| 久久SE精品一区二区| 精品无码久久久久久久久久| 久久亚洲精品国产精品| 久久天天躁狠狠躁夜夜av浪潮| 久久国产乱子伦免费精品| 2019久久久高清456| 久久久久久青草大香综合精品| 日韩精品久久久久久久电影蜜臀| 无码人妻少妇久久中文字幕| 国产精品丝袜久久久久久不卡| 亚洲中文久久精品无码ww16| 久久久精品久久久久久| 91性高湖久久久久| 久久综合九色综合久99| 久久国产精品无码一区二区三区| 伊人久久成人成综合网222| 曰曰摸天天摸人人看久久久| 97超级碰碰碰久久久久| 久久久国产打桩机| 综合久久国产九一剧情麻豆 | 国内精品久久久久久中文字幕| 久久婷婷激情综合色综合俺也去| 久久精品国产亚洲AV香蕉|