OC基本數(shù)據(jù)類型

對于開發(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:

NSInteger&NSUInteger的實質(zhì).png

根據(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位,分布如下:

浮點數(shù)據(jù)分布.png

根據(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位。

這樣就得到如下表格:

基本數(shù)據(jù)類型取值范圍.png

歡迎交流.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容