對于開發(fā)者來說,像基本數(shù)據(jù)類型這種比較基礎(chǔ)的知識應(yīng)該是不被擔(dān)憂的,但是前段時間檢查項目代碼時發(fā)現(xiàn)居然有tag越界的情況。
簡單說一下tag越界的危害:
viewWillAppear:方法中展示view,并給view一個tag值;viewWillDisappear:方法中remove這個view,假設(shè)這里必須通過tag值來找到需要被remove的view。
一旦tag越界,那么在remove的時候,我們就無法找到目標(biāo)view。如果反復(fù)進(jìn)入這個頁面的話,那么就會只有創(chuàng)建,沒有釋放。視覺上:視圖重復(fù);內(nèi)存上:消耗增加。
當(dāng)然,這里還需要注意的是,對基本數(shù)據(jù)類型的賦值一定要考慮當(dāng)前系統(tǒng)是32位的,還是64的。開發(fā)必須嚴(yán)謹(jǐn),不能走大眾路線。
數(shù)據(jù)類型取值范圍還是看所占內(nèi)存大小,首先分別在32位和64位機(jī)器上打印一下它們所占的內(nèi)存大小:
NSLog(@"char: %lu",sizeof(char));
NSLog(@"int:%lu",sizeof(int));
NSLog(@"short:%lu",sizeof(short));
NSLog(@"long: %lu",sizeof(long));
NSLog(@"unsigned int:%lu",sizeof(unsignedint));
NSLog(@"unsigned short:%lu",sizeof(unsignedshort));
NSLog(@"unsigned long: %lu",sizeof(unsignedlong));
NSLog(@"long long:%lu",sizeof(longlong));
NSLog(@"NSInteger: %lu",sizeof(NSInteger));
NSLog(@"NSUInteger:%lu",sizeof(NSUInteger));
/*
32位機(jī)器打印結(jié)果:
char: 1
int: 4
short:2
long: 4
unsigned int: 4
unsigned short:2
unsigned long: 4
long long: 8
NSInteger: 4
NSUInteger:4
*/
/*
64位機(jī)器打印結(jié)果:
char: 1
int: 4
short:2
long: 8
unsigned int: 4
unsigned short:2
unsigned long: 8
long long: 8
NSInteger: 8
NSUInteger:8
*/
基于計算機(jī)的二進(jìn)制計數(shù)法,這里來看一下計算規(guī)則:1Byte = 8bit,對于無符號數(shù):沒有符號位,也就是8位全部可以表示數(shù)據(jù),并且最小值是0,取值范圍:0 ~ 28;對于有符號數(shù):1位符號位,這樣只有7位用來表示數(shù)據(jù),取值范圍:-27 ~ 2^7-1;然后分別在32位和64位機(jī)器上打印一下它們的最大最小值:
int max_int = pow(2, 31) - 1;
int min_int = - pow(2, 31);
NSLog(@"int: max_int:%d min_int:%d", max_int, min_int);
//32:int: max_int:2147483647 min_int:-2147483648
//64:int: max_int:2147483647 min_int:-2147483648
short max_short = pow(2, 15) - 1;
short min_short = - pow(2, 15);
NSLog(@"short: max_short:%d min_short:%d", max_short, min_short);
//32:short: max_short:32767 min_short:-32768
//64:short: max_short:32767 min_short:-32768
long max_long = powl(2, 63) - 1;
long min_long = - powl(2, 63);
NSLog(@"long: max_long:%ld min_long:%ld", max_long, min_long);
//32:long: max_long:2147483647 min_long:-2147483648
//64:long: max_long:9223372036854775807 min_long:-9223372036854775808
unsigned int max_uint = pow(2, 32) - 1;
unsigned int min_uint = - pow(2, 32);
NSLog(@"unsigned int: max_uint:%u min_uint:%u", max_uint, min_uint);
//32:unsigned int: max_uint:4294967295 min_uint:0
//64:unsigned int: max_uint:4294967295 min_uint:0
unsigned short max_ushort = pow(2, 16) - 1;
unsigned short min_ushort = - pow(2, 16);
NSLog(@"unsigned short: max_ushort:%u min_ushort:%u", max_ushort, min_ushort);
//32:unsigned short: max_ushort:65535 min_ushort:0
//64:unsigned short: max_ushort:65535 min_ushort:0
unsigned long max_ulong = powl(2, 64) - 1;
unsigned long min_ulong = - powl(2, 64);
NSLog(@"unsigned long: max_ulong:%lu min_ulong:%lu", max_ulong, min_ulong);
//32:unsigned long: max_ulong:4294967295 min_ulong:0
//64:unsigned long: max_ulong:18446744073709551615 min_ulong:9223372036854775808
long long max_llong = powl(2, 63) - 1;
long long min_llong = - powl(2, 63);
NSLog(@"long long: max_llong:%lld min_llong:%lld", max_llong, min_llong);
//32:long long: max_llong:9223372036854775807 min_llong:-9223372036854775808
//64:long long: max_llong:9223372036854775807 min_llong:-9223372036854775808
NSInteger max_integer = powl(2, 63) - 1;
NSInteger min_integer = - powl(2, 63);
NSLog(@"NSInteger: max_integer:%ld, min_integer:%ld", max_integer, min_integer);
//32:NSInteger: max_integer:2147483647, min_integer:-2147483648
//64:NSInteger: max_integer:9223372036854775807, min_integer:-9223372036854775808
NSUInteger max_uinteger = powl(2, 64) - 1;
NSUInteger min_uinteger = - powl(2, 64);
NSLog(@"NSUInteger: max_uinteger:%lu, min_uinteger:%lu", max_uinteger, min_uinteger);
//32:NSUInteger: max_uinteger:4294967295, min_uinteger:0
//64:NSUInteger: max_uinteger:18446744073709551615, min_uinteger:9223372036854775808
注:展開NSInteger和NSUInteger后發(fā)現(xiàn),他們分別對應(yīng)int和unsigned int:

根據(jù)輸出結(jié)果發(fā)現(xiàn)unsigned long(NSUInteger)類型值的最小值并不是0。這是因為無符號長整型數(shù)一旦給了負(fù)值,會按無符號數(shù)字來處理。
例如-1,會被默認(rèn)成沒有符號位的所有二進(jìn)制位均為1,那么打印的值為:18446744073709551615。
例如二進(jìn)制數(shù)1111表示15,如果最多用4位二進(jìn)制數(shù)表示-pow(2, 4),寫法為: -(1111+1)。最多4位2進(jìn)制數(shù)的計算過程:-(1111+1) --> -(0000) --> 1000,那么此時這個數(shù)為十進(jìn)制的8。
所以- powl(2, 64)的64位二進(jìn)制數(shù)全為1的值為18446744073709551615,那么powl(2, 64)的二進(jìn)制值就是64位均為0的二進(jìn)制數(shù),那么-powl(2, 64)的二進(jìn)制數(shù)為第一位為1,剩余63位均為0的二進(jìn)制數(shù),轉(zhuǎn)化成十進(jìn)制數(shù)值為:9223372036854775808。
結(jié)論:這里的輸出值沒有錯,但不是實際意義上的最小值,無符號長整型的最小值應(yīng)該是0。
另外對于float和double存儲空間的打印,32位和64位均為:
NSLog(@"float:%lu", sizeof(float));
NSLog(@"double:%lu", sizeof(double));
//float:4
//double:8
對于float和double的范圍是由指數(shù)的位數(shù)來決定的。
float的指數(shù)位有8位,而double的指數(shù)位有11位,分布如下:

根據(jù)上表可以看到,float的指數(shù)范圍為-127+128,而double的指數(shù)范圍為-1023+1024,并且指數(shù)位是按補(bǔ)碼的形式來劃分的。其中負(fù)指數(shù)決定了浮點數(shù)所能表達(dá)的絕對值最小的非零數(shù);而正指數(shù)決定了浮點數(shù)所能表達(dá)的絕對值最大的數(shù),也即決定了浮點數(shù)的取值范圍。
float的范圍為-2^128 ~ +2^128,也即-3.40E+38 ~ +3.40E+38;double的范圍為-2^1024 ~ +2^1024,也即-1.79E+308 ~ +1.79E+308。
這里還要說下float和double的精度:
float和double的精度是由尾數(shù)的位數(shù)來決定的。浮點數(shù)在內(nèi)存中是按科學(xué)計數(shù)法來存儲的,其整數(shù)部分始終是一個隱含著的“1”,由于它是不變的,故不能對精度造成影響。
float:2^23 = 8388608,一共七位,這意味著最多能有7位有效數(shù)字,但絕對能保證的為6位,也即float的精度為6~7位有效數(shù)字;
double:2^52 = 4503599627370496,一共16位,同理,double的精度為15~16位。
這樣就得到如下表格:

歡迎交流.