unsigned 整型实现无溢出运算

  普通的 int 整型能表示的范围很有限,所以刷题时很多时候不得不用 long long 来存更大的数据。或者找出数列中某个只出现一次(或奇数次)的数(其余的数均出现两次 / 偶数次),用异或运算的经典题目,然而,如果数据范围不大的话,O(n) 的复杂度下,把第一次出现的所有数全加 / 乘起来,然后再减 / 除去第二次出现的 n - 1 个数(缺少的那个就是要求的数),原理是可以的,只是一般 OJ 出这样的题目必然不会让你用这种小学生的思维去做,所以会把数据设置得奇大,让你一累加必然溢出,更不用说乘了;然而,我发现,用无符号的整数貌似能避免这样的问题,因为它溢出后自动环回,就像自动取模一样,比如 unsigned int,对于溢出的运算,都会将结果对 1<<32 取模(实际上我也解释不清楚,组成原理没学好 o(╯□╰)o ),我写了个类似的程序发现好像真的是这样,虽说没有严格测试过:

 1 #include<cstdio>
 2 #include<cstring>
 3 typedef unsigned int ui;
 4 const ui inf = 0x3fffffff;
 5 
 6 int main() {
 7     ui sum = 0, n, x;
 8     printf("inf = %u\n",inf);
 9 
10     n = 300;
11     while(n--) {
12         sum += inf;
13         sum += inf / 2;
14         sum += inf - inf / 3;
15     }
16     printf("sum = %u\n",sum);
17 
18     for(int i = 1; i <= 300; ++i) {
19         sum -= inf - inf / 3;
20         sum -= inf / 2;
21         sum -= inf;
22     }
23     printf("%u\n",sum);
24 
25     return 0;
26 }
View Code

  运行结果如下:

posted @ 2015-08-01 10:59  Newdawn_ALM  阅读(494)  评论(0编辑  收藏  举报