c語(yǔ)言算術(shù)運(yùn)算符越界問題解決方案
作為一個(gè)系統(tǒng)程序員, 有必要對(duì)這些細(xì)節(jié)有深入的了解. 本篇參考csapp, 主要介紹如何判斷算術(shù)運(yùn)算的越界問題.
(雖然本篇的代碼經(jīng)過大量的測(cè)試, 但本人仍然無(wú)法保證代碼的正確性, 希望大家糾錯(cuò)).
講解的原則是"擺定理, 不證明, 寫代碼". 具體的證明過程在csapp中有詳細(xì)的講解, 也不是太難. 主要使用關(guān)鍵定理來(lái)寫代碼. Go~
問題一: 無(wú)符號(hào)數(shù)的加法越界問題
[定理]

[理解]
這個(gè)定理比較容易, 也比較能讓人接受. 不解釋啦.
/* Determine whether arguments can be added without overflow */
int uadd_ok(unsigned int x, unsigned int y)
{
return !(x+y < x);
}
問題二: 無(wú)符號(hào)數(shù)的減法越界問題
[定理]


[理解]
1. 計(jì)算機(jī)中沒有減法, x-y = x+(-y), 這里的-y就是上述的y的加法逆元. 不管是有符號(hào)還是無(wú)符號(hào), 都是轉(zhuǎn)換為加法運(yùn)算. 只是加法逆元的定義不同. 
3. C語(yǔ)言保證 -x = ~x+1; 可以驗(yàn)證這種方式與上面公式等價(jià).
4. s=x-y = x+(-y). 那么 不會(huì)溢出 等價(jià)于 y不為0 或者 !uadd_ok(x, -y).

/* Determine whether argumnts can be substracted without overflow */
int usub_ok(unsigned int x, unsigned int y)
{
return !y || !uadd_ok(x, -y);
}
問題三: 無(wú)符號(hào)數(shù)的乘法越界問題
[定理]

[理解]
等價(jià)條件可以相互推導(dǎo)即可.
/* Determine whether arguments can be multiplied without overflow */
int umul_ok(unsigned int x, unsigned int y)
{
unsigned int p = x * y;
return !x || p/x==y;
}
問題四: 有符號(hào)數(shù)的加法越界問題
[定理]
對(duì)于兩個(gè)有符號(hào)數(shù)x, y. 越界的等價(jià)條件是x,y為負(fù)數(shù), x+y為正數(shù)或者x,y為正數(shù), x+y為負(fù)數(shù).
[理解]
這個(gè)定理比較容易.
/* Determine whether arguments can be added without overflow */
int tadd_ok(int x, int y)
{
return !(x<0&&y<0&&x+y>0 || x>0&&y>0&&x+y<0);
}
問題五: 有符號(hào)數(shù)的減法越界問題
[定理]


[理解]
同無(wú)符號(hào)的減法一樣, 只是加法逆元的定義不同, 但是位模式是一樣的. C語(yǔ)言可以保證-x=~x+1. 同樣也分兩種情況討論.見代碼.
/* Determine whether arguments can be subtracted without overflow */
int tsub_ok(int x, int y)
{
#if 0
if (y == INT_MIN)
return x<0;
else
return tadd_ok(x, -y);
#endif
return y==INT_MIN&&x<0 || y!=INT_MIN&&tadd_ok(x, -y);
}
問題六: 有符號(hào)數(shù)的乘法越界問題
[定理]
完全同無(wú)符號(hào)的乘法一樣.
/* Determine whether arguments can be multiplied without overflow. */
int tmul_ok(int x, int y)
{
#if 0
int p = x * y;
return !x || p/x==y;
#endif
return umul_ok(x, y); /* 直接調(diào)用 */
}
- C語(yǔ)言運(yùn)算符優(yōu)先級(jí)列表(超詳細(xì))
- C語(yǔ)言位運(yùn)算符:與、或、異或、取反、左移與右移詳細(xì)介紹
- C語(yǔ)言 解決不用+、-、×、÷數(shù)字運(yùn)算符做加法的實(shí)現(xiàn)方法
- C語(yǔ)言中6組指針和自增運(yùn)算符結(jié)合方式的運(yùn)算順序問題
- C語(yǔ)言運(yùn)算符及其優(yōu)先級(jí)匯總表口訣
- C語(yǔ)言中邏輯運(yùn)算符與條件運(yùn)算符的學(xué)習(xí)教程
- 簡(jiǎn)單總結(jié)C語(yǔ)言中的運(yùn)算符優(yōu)先級(jí)
- C語(yǔ)言運(yùn)算符的優(yōu)先級(jí)和結(jié)合性實(shí)例詳解
- C語(yǔ)言邏輯運(yùn)算符知識(shí)整理
- C語(yǔ)言 運(yùn)算符詳細(xì)介紹及示例代碼
相關(guān)文章
c語(yǔ)言中的二級(jí)指針做函數(shù)參數(shù)說(shuō)明
這篇文章主要介紹了c語(yǔ)言中的二級(jí)指針做函數(shù)參數(shù)說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05
C++中AVL樹的底層以及實(shí)現(xiàn)方法總結(jié)
這篇文章主要介紹了C++中AVL樹的底層以及實(shí)現(xiàn)方法的相關(guān)資料,AVL樹是一種自平衡的二叉搜索樹,每個(gè)節(jié)點(diǎn)的左右子樹高度差不超過1,通過旋轉(zhuǎn)操作保持平衡,詳解了AVL樹的結(jié)構(gòu)、插入、旋轉(zhuǎn)、查找和遍歷方法,展示了其保持平衡的機(jī)制及對(duì)應(yīng)代碼實(shí)現(xiàn),需要的朋友可以參考下2024-10-10
生成隨機(jī)數(shù)rand函數(shù)的用法詳解
本篇文章是對(duì)生成隨機(jī)數(shù)rand函數(shù)的用法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
《C++ Primer》隱式類類型轉(zhuǎn)換學(xué)習(xí)整理
在本篇文章里小編給大家整理的是關(guān)于《C++ Primer》隱式類類型轉(zhuǎn)換學(xué)習(xí)筆記內(nèi)容,需要的朋友們參考下。2020-02-02
深入理解約瑟夫環(huán)的數(shù)學(xué)優(yōu)化方法
本篇文章是對(duì)約瑟夫環(huán)的數(shù)學(xué)優(yōu)化方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
Qt自繪實(shí)現(xiàn)蘋果按鈕滑動(dòng)效果的示例代碼
這篇文章主要介紹了Qt自繪實(shí)現(xiàn)蘋果按鈕滑動(dòng)效果的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
C++中常見容器類的使用方法詳解(vector/deque/map/set)
C++中常見的容器類有vector、list、deque、map、set、unordered_map和unordered_set。下面將舉例直接說(shuō)明各個(gè)容器的使用方法,希望對(duì)大家有所幫助2023-03-03

