青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

C++之竹

無論是太陽下,還是風雨中,都要成長!

常用鏈接

統(tǒng)計

最新評論

不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算

最近,在看《劍指Offer——名企面試官精講典型編程題》一書,當看到“面試題47:不用加減乘除做加法”時,經(jīng)過一分鐘左右思考后,得出了思路,跟書上一對照,基本一致,呵呵O(∩_∩)O~。于是,隨即又開始思考:加法是實現(xiàn)了,那么減法、乘法還有除法又該怎么實現(xiàn)呢?一番思考與分析后,得出算法,寫出代碼,測試通過,Happy??!\(^o^)/~

 接下來,就將我的算法與代碼展示如下,還請有更好算法的牛人不吝賜教?。?/p>

 

【1. 加法】

 因本人的算法與《劍指Offer——名企面試官精講典型編程題》中一致,就不做贅述,而只貼出代碼。

代碼1.1 加法:計算a+b

int Add(int a, int b)
{
    
int sum = a ^ b;
    
int carry = a & b;
    
while (carry != 0{
        a 
= sum;
        b 
= carry << 1;
        sum 
= a ^ b;
        carry 
= a & b;
    }


    
return sum;
}

【2. 減法】

方案一:通過Add函數(shù)的實現(xiàn)

按常規(guī)思路,根據(jù)加減運算的互反性(即,減去一個數(shù)等于加上這個數(shù)的相反數(shù)),然后利用前面已實現(xiàn)的Add函數(shù)來進行實現(xiàn)。

按這個思路,我們首先要做到對一個整數(shù)取相反數(shù),不能使用運算符“-”,那么就只能根據(jù)C++上兩個互為相反數(shù)的int型數(shù)據(jù)的二進制結(jié)構(gòu)關(guān)系——整數(shù)的相反數(shù)等于該數(shù)按位取反再加1,來設(shè)計如下的函數(shù)了。

代碼2.1 取整數(shù)n的相反數(shù)

int OppositeNumber(int n)
{
    
return Add(~n, 1);
}

然后就可以通過Add函數(shù)來實現(xiàn)減法了。

代碼2.2 減法一:計算a-b

int Subtract(int a, int b)
{
    
return Add(a, OppositeNumber(b));
}

畢竟是作為對思維開放的一個鍛煉,所以對于直接的減法算法的思考還是值得的。于是就有了下面的方案二。

方案二:直接通過位操作實現(xiàn)減法

算法是得通過二進制位計算來實現(xiàn)的,所以在分析減法時從二進制減法計算的角度去考慮將更合適。那么,兩個二進制形式整數(shù)的減法操作又是怎樣進行的呢?

  1. 首先,如果減數(shù)為0,則被減數(shù)即為減法的結(jié)果,運算結(jié)束。
    但如果減數(shù)不為0,我們可以先把被減數(shù)和減數(shù)上同為1的位從兩個數(shù)上去除。至于如何分離出值同為1的位,則可以通過求位與操作來做到;而把這些1分別中被減數(shù)和減數(shù)中去除,則可以通過按位異或來的操作來實現(xiàn)。
  2. 經(jīng)步驟1處理后,被減數(shù)和減數(shù)在對應(yīng)的位上,將或者通為0,或者分別為0和1,卻不會同為1。此時:
    如果對應(yīng)位被減數(shù)=1,而減數(shù)=0,則所得結(jié)果對應(yīng)位也為1;
    如果對應(yīng)位被減數(shù)=0,而減數(shù)=1,則所得結(jié)果對應(yīng)位還是1,但此時須向前一位借1;
    即,通過被減數(shù)與減數(shù)作位異或的操作得到臨時結(jié)果,和通過對減數(shù)左移一位得到需從臨時結(jié)果中減去的借數(shù)。
  3. 于是,經(jīng)過步驟2后,原來的減法變成了要求:臨時結(jié)果 - 借數(shù)
    很明顯,只要以臨時結(jié)果為被減數(shù),借數(shù)為減數(shù),重復步驟1~3即可。

上述步驟中,如果被減數(shù)或減數(shù)為負數(shù),由負數(shù)的二進制位結(jié)構(gòu),可以保證上述步驟的處理仍適用,證明過程就請恕我在這里略去了。具體的實現(xiàn)代碼如下。

代碼2.3 減法二:計算a-b

int Subtract(int a, int b)
{
    
while (b != 0
    
{
        
int sameBits = a & b;
        a 
^= sameBits;
        b 
^= sameBits;
        a 
|= b;
        b 
<<= 1;
    }


    
return a;
}

※注:上述加法和減法中,按代碼安全性,其實還應(yīng)考慮計算后數(shù)據(jù)溢出的情況,這里我偷了下懶,省去了。不過下面的乘除法,我會提供包含了異常處理的代碼。異常處理的方式,我采用了throw拋出的方式。

 【3. 乘法】

為了方便對數(shù)據(jù)溢出的統(tǒng)一處理,在進行計算前,我先保存了被乘數(shù)與乘數(shù)的符號信息,并當被乘數(shù)或乘數(shù)為負時,利用上面的OppositeNumber函數(shù),統(tǒng)一的轉(zhuǎn)換為正整數(shù)(或0),然后再來進行乘法的運算。為了能同時適應(yīng)32位和64位的整形數(shù),在取符號信息與設(shè)置溢出判斷位時,使用了以下的輔助宏和函數(shù)。

代碼3.1 輔助宏與輔助函數(shù)

#define BITS_OF_ONE_BYTE     8
#define SIGN_BIT_FLAG_FOR_BYTE   0x80     // for signed byte/char

int SignBitFlagOfInt()
{
    
static int signBitFlag = 0;

    
static bool bIs1stCalled = true;
    
if (bIs1stCalled)
    
{
        
int temp = SIGN_BIT_FLAG_FOR_BYTE;
        
while (temp != 0
        
{
            signBitFlag 
= temp;
            temp 
<<= BITS_OF_ONE_BYTE;
        }

        bIs1stCalled 
= false;
    }


    
return signBitFlag;
}

乘法的算法??紤]到:

  1. 整數(shù)n乘以2k == n << k
  2. C++中的任何一個非負int型數(shù)據(jù)都可以表示為如下的形式:
    k0×20+k1×21+...+km×2m
    的形式。(其中:ki∈{0, 1}, i∈{0, 1, ... , m}, 32位int型m = 30, 64位int型m = 62)

于是,就可以利用乘法分配率,通過循環(huán)累加,進行乘法的運算了。參考代碼3.2。

代碼3.2 乘法:計算a×b

int Multiply(int a, int b)
{
    
int signStatA = a & SignBitFlagOfInt();
    
if (signStatA != 0)
    
{
        a 
= OppositeNumber(a);
    }


    
int signStatB = b & SignBitFlagOfInt();
    
if (signStatB != 0)
    
{
        b 
= OppositeNumber(b);
    }


    
int overFlowFlag = SignBitFlagOfInt(); // used for checking if the signed int data overflowed.
    int product = 0// the result of |a| * |b|
    for (int curBitPos = 1; curBitPos != 0; curBitPos <<= 1)
    
{
        
if ((b & curBitPos) != 0)
        
{
            product 
= Add(product, a);
            
if (((a & overFlowFlag) != 0
                
|| ((product & overFlowFlag) != 0))
            
{
                
throw std::exception("The result data war overflowed.");
            }

        }

        a 
<<= 1;
    }


    
return ((signStatA ^ signStatB) == 0? product : OppositeNumber(product);
}

【4. 除法】

整數(shù)的除法,不同于乘法,除法所得的商的絕對值必然不大于被除數(shù)的絕對值,而所得余數(shù)的絕對值則必然小于除數(shù)的絕對值。所以,在設(shè)計除法函數(shù)的時候,無需考慮數(shù)據(jù)溢出的問題。但對于除法,卻也有它自己的禁忌——除數(shù)不能為“0”。

為了處理的方便,準備工作同乘法一樣,記錄下被除數(shù)與除數(shù)的符號狀態(tài)(比便在計算出結(jié)果后進行符號的調(diào)整),并當被除數(shù)或除數(shù)為負時,通過函數(shù)OppositeNumber將其轉(zhuǎn)換為相反數(shù)。于是,接下來,我就只需考慮“非負整數(shù)(>=0)÷正整數(shù)(>0)”的情況了。對這種情況,計算過程如下:

  1. 預(yù)備工作:置商為0;
  2. 判斷“被除數(shù)>=除數(shù) ”是否成立:
    成立,繼續(xù)步驟3;
    不成立,被除數(shù)的值賦給余數(shù),計算結(jié)束。
  3. 備份除數(shù),并設(shè)置商分子(一個臨時變量,最終需加到商上面,故暫且如此命名)為1;
    對商分子和除數(shù)同步向左移位,直到繼續(xù)移位將大于被除數(shù)時為止;
  4. 從被除數(shù)上減去除數(shù),并將商加上商分子。
  5. 通過備份的除數(shù)值還原除數(shù),跳轉(zhuǎn)到步驟2繼續(xù)執(zhí)行。

對應(yīng)的代碼參加代碼4.1。

代碼4.1 除法:計算a÷b 

int Divide(int a, int b, int * pRem /*= NULL*/)
{
    
if (b == 0)
    
{
        
throw std::exception("Invalid divisor!! (The divisor can't be 0!)");
    }


    
int signStatA = a & SignBitFlagOfInt();
    
if (signStatA != 0)
    
{
        a 
= OppositeNumber(a);
    }


    
int signStatB = b & SignBitFlagOfInt();
    
if (signStatB != 0)
    
{
        b 
= OppositeNumber(b);
    }


    
int quotient = 0;
    
int backupB = b;
    
while (a >= b) 
    
{
        
int tempB = b << 1;
        
int tempQ = 1;
        
while ((tempB <= a) && ((tempB & SignBitFlagOfInt()) == 0))
        
{
            b 
= tempB;
            tempQ 
<<= 1;
            tempB 
<<= 1;
        }


        a 
= Subtract(a, b);
        quotient 
|= tempQ;
        b 
= backupB;
    }


    
if (pRem != NULL)
    
{
        
*pRem = a;
    }


    
if ((signStatA != 0&& (a != 0))
    
{
        quotient 
= Add(quotient, 1);
        
if (pRem != NULL)
        
{
            
*pRem = Subtract(b, *pRem);
        }

    }


    
return ((signStatA ^ signStatB) == 0? quotient : OppositeNumber(quotient);
}

※注:函數(shù)的返回值即為所求的商;
       參數(shù)pRem為余數(shù)的傳出參數(shù),其默認值NULL,表示當前無需關(guān)注余數(shù)。

 

posted on 2012-03-30 01:30 青碧竹 閱讀(5009) 評論(7)  編輯 收藏 引用 所屬分類: 算法相關(guān)

評論

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-03-30 13:02 遠行

搜了下那本書,發(fā)現(xiàn)好像沒出版,lz有那本書的文檔能不能發(fā)一份啊,
愛好書之情難自抑啊。。。  回復  更多評論   

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-03-30 14:40 FiO

@遠行
http://book.douban.com/subject/6966465/  回復  更多評論   

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-03-30 18:26 遠行

沒錢買書,只看電子版。。。@FiO
  回復  更多評論   

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-03-30 19:24 flyinghearts


嚴格的講,這些方法只能用于無符號數(shù)。對有符號數(shù)這樣的做法是不通用的。 google 一補數(shù)。

  回復  更多評論   

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-03-30 19:49 青碧竹

@flyinghearts
在目前計算機內(nèi)負整數(shù)通過對其絕對值進行求反+1來存儲的前提下,這些算法完全適用于有符號整數(shù)。  回復  更多評論   

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-03-31 13:15 SwordLance

這類問題,只要知道CPU的運算器是怎么做的,用程序模擬即可。  回復  更多評論   

# re: 不使用 +-×÷ 運算符來實現(xiàn) 加減乘除 四項運算 2012-04-01 00:13 青碧竹

@SwordLance
是的。  回復  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精一区二区三区| 国语自产精品视频在线看抢先版结局 | 亚洲欧美日韩成人| 欧美黄色小视频| 激情婷婷久久| 久久久久久夜精品精品免费| 日韩视频专区| 欧美大色视频| 亚洲精品视频在线| 欧美成人激情视频免费观看| 欧美在线观看日本一区| 欧美日韩精品福利| 99国产精品久久久久久久| 欧美激情 亚洲a∨综合| 蜜桃av一区二区三区| 91久久国产综合久久91精品网站| 美女脱光内衣内裤视频久久影院| 欧美在线视频免费观看| 狠狠色综合日日| 免费观看成人网| 久久综合五月天婷婷伊人| 亚洲电影在线播放| 亚洲高清不卡在线观看| 久久夜色精品| 亚洲精品久久久蜜桃| 亚洲美女中出| 国产精品自拍三区| 另类天堂av| 欧美一区二区三区成人| 西瓜成人精品人成网站| 亚洲欧美日韩爽爽影院| 韩国欧美一区| 欧美激情第9页| 欧美日韩视频不卡| 欧美伊人精品成人久久综合97 | 欧美日韩第一页| 亚洲少妇自拍| 欧美在线视频一区二区三区| 在线观看日韩一区| 亚洲人成啪啪网站| 国产精品人人爽人人做我的可爱| 久久久www成人免费精品| 久久久久九九视频| 亚洲私人影院在线观看| 欧美在线欧美在线| 一区二区激情小说| 欧美一区二区黄| 亚洲六月丁香色婷婷综合久久| 中国成人亚色综合网站| 在线成人h网| 亚洲一区二区三区乱码aⅴ蜜桃女| 国内一区二区在线视频观看| 最新国产拍偷乱拍精品| 国产日韩精品电影| 亚洲人成久久| 国产婷婷成人久久av免费高清| 欧美激情亚洲国产| 国产日韩av一区二区| 日韩图片一区| 亚洲国产精品va在线看黑人| 一区二区三区鲁丝不卡| 亚洲激情在线| 久久精品国产91精品亚洲| 一区二区三区视频免费在线观看| 久久精品国产v日韩v亚洲| 亚洲亚洲精品三区日韩精品在线视频| 欧美怡红院视频一区二区三区| 中文精品视频| 欧美国产日本在线| 欧美成人一区在线| 国产一区二区日韩| 亚洲永久免费精品| av成人免费观看| 免费成人高清在线视频| 欧美中文字幕在线观看| 欧美色图麻豆| 亚洲精品综合久久中文字幕| 亚洲国产欧美一区| 久热精品视频在线观看| 玖玖视频精品| 极品中文字幕一区| 久久久久九九九九| 美女精品国产| 在线观看av一区| 久久久久久久网站| 久久在精品线影院精品国产| 国产日韩欧美在线视频观看| 亚洲资源av| 欧美一区二区视频免费观看| 欧美性猛交xxxx乱大交蜜桃| 日韩天堂在线观看| 亚洲午夜视频在线观看| 欧美午夜激情小视频| 亚洲精品视频免费观看| 欧美日韩国产综合视频在线观看| 久久久99国产精品免费| 国产欧美日韩视频| 亚洲欧美日韩天堂| 久久国产精品99精品国产| 国产欧美在线看| 久久精品亚洲精品国产欧美kt∨| 久久精品国产清自在天天线 | 一区二区三区高清视频在线观看 | 亚洲欧美久久久久一区二区三区| 欧美日韩美女一区二区| 夜夜嗨av一区二区三区网页| 亚洲香蕉伊综合在人在线视看| 国产精品jvid在线观看蜜臀| 亚洲一区二区三区四区视频| 欧美在线视频二区| 国产主播精品| 另类图片国产| 日韩网站在线| 久久国产毛片| 亚洲高清视频中文字幕| 欧美日韩美女在线| 亚洲欧美成人一区二区在线电影| 久久精品国产亚洲一区二区| 亚洲大片免费看| 欧美日韩国产色视频| 亚洲女ⅴideoshd黑人| 欧美综合国产精品久久丁香| 尤物视频一区二区| 欧美日韩成人在线观看| 欧美一区二区三区免费视| 亚洲国产精品一区二区三区| 亚洲视频网在线直播| 国产一区二区三区最好精华液| 欧美成人午夜激情视频| 亚洲一区久久久| 欧美国产欧美亚州国产日韩mv天天看完整| 亚洲国产乱码最新视频| 国产精品国产三级国产aⅴ无密码 国产精品国产三级国产aⅴ入口 | 欧美1区3d| 亚洲午夜在线视频| 免费观看在线综合色| 亚洲一区二区免费视频| 黄色成人av网站| 欧美日韩国产综合在线| 久久久久久午夜| 亚洲影视在线| 亚洲人成在线观看一区二区 | 午夜在线观看欧美| 亚洲区第一页| 免费不卡在线观看| 亚洲一区一卡| 亚洲精选在线| 怡红院精品视频在线观看极品| 欧美午夜激情小视频| 欧美福利视频| 久久亚洲春色中文字幕| 午夜在线a亚洲v天堂网2018| 99在线热播精品免费99热| 欧美激情女人20p| 久久麻豆一区二区| 精品999网站| 久久九九国产精品怡红院| 亚洲开发第一视频在线播放| 久久亚洲电影| 久久成人综合视频| 亚洲欧美不卡| 亚洲天堂av在线免费观看| 亚洲精品一区二区三区不| 国内精品美女av在线播放| 国产精品一区二区在线| 欧美视频免费看| 欧美日韩国产在线播放| 欧美韩日一区二区| 美女精品视频一区| 久久精品视频在线看| 午夜精品在线视频| 在线视频你懂得一区二区三区| 亚洲片在线资源| 亚洲乱码久久| 亚洲靠逼com| 日韩小视频在线观看| 亚洲毛片在线| av成人免费| 亚洲天堂男人| 亚洲女同性videos| 亚洲一区影院| 亚洲欧美精品suv| 久久成人精品电影| 久久九九99视频| 久久久精彩视频| 麻豆91精品| 欧美国产国产综合| 亚洲欧洲在线观看| 一区二区三区黄色| 欧美亚洲一区二区三区| 欧美尤物一区| 欧美ab在线视频| 欧美喷潮久久久xxxxx| 欧美三级日本三级少妇99| 欧美性感一类影片在线播放| 国产精品一区免费观看| 激情成人在线视频| 亚洲美女在线国产| 亚洲一区二区三区涩| 欧美在线视频一区二区|