您好,登錄后才能下訂單哦!
本篇文章給大家分享的是有關C++中怎么實現類型轉換,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
在進行下面的學習前,我覺得有比較知道不同類型是怎么進行轉換的。
int a = 777777; //二進制為00000000 01110110 10101101 11110001 short b = a; //b只有2字節,從低位開始截斷,只能存10101101 11110001 注意,這并不是b的最終答案, //我們應該知道,計算機中數值都是以二進制補碼的形式存儲的,所以我們需要將10101101 11110001還原為原碼, //也就是11011110 00110001,最高位是符號位,轉換為十進制就是-8655 cout << b;
而浮點數轉整形,不但會進行上述過程還會進行小數截斷。
int int_a = 123; long long int llong_a = int_a; //賦值的時候,編譯器會先將int類型的123擴展為long類型123的新值,然后賦值給long_b,原先的int_a還是int類型, //沒有變化。 cout << "llong_b所占內存: " << sizeof(llong_a)<< " 值為: " << llong_a << " int_a的類型:" << sizeof(int_a) << endl; //通常情況下,小范圍轉大范圍這樣賦值是沒有問題,但是如果大范圍轉小范圍可能回來帶來一些麻煩, //如果大范圍的數值在小范圍之內,這也是沒有問題的,如果該數值不在小范圍之內會發生什么呢 //long long int 最大值為9223372036854775807 //而int的最大值為2147483647 我們來做個實驗: long long int llong_b = 9223372036854775807; int int_b = llong_b; cout << "\nllong_b所占內存: " << sizeof(int_b) << " 值為: " << int_b << " int_a的類型:" << sizeof(llong_b) << endl; //出現了問題,int_b的值只有-1,連自己本身類型的最大值都沒有賦到。
運行結果:
小范圍類型賦值給大范圍類型是可以的,大范圍賦值給小范圍,要考慮好是否超出最大值,通常只會復制低位,建議不要這樣做。
上面說的是整形類型的轉換,如果是浮點數轉換的話也會有兩個問題:
1.將較大的浮點型轉換為較小的浮點類型,精度降低(如果對精度不理解請看我的C++第一篇),值可能會超出目標類型的取值范圍,這種情況下的值是不確定的。
2.將浮點型轉換為整形,小數部分會被截斷,原來的值可能超出目標類型的取值范圍,這種情況下的值也是不確定的。
用{}這種方式來轉換類型是C++11新增的內容,它更為嚴格,不允許需要轉換的類型進行縮窄,什么意思呢,就是要保證涉及到需要轉換的類型應該是和需要完成的類型應該是一樣長的,比如,int有4位,long long 有8位,long long想轉為int,就必須將從左往右的4位截斷,這就是縮窄。
const int code = 66; int x = 66; char c1{ 31325 }; //錯誤 char c2 = { 66 }; char c3 = { code }; char c4 = { x }; // 錯誤 x = 31325; char c5 = x;
代碼的語法沒有任何問題,但是編譯運行時會出現:
第一個錯誤好理解一點,31325遠遠超過了char的最大范圍。
第二個錯誤明明x的值為66,為什么會出錯呢?編譯器不會管你x的值是多大,他只管x的類型是多大。
而最后c5被賦予31325這個值,由于沒有使用{}處理,并沒有保存,但其結果是不確定的。
而浮點數轉為整形,即使符合也不被允許:
long long int a = {10.12f}; long long int b = { 10.12 };
而整形轉浮點數,只要符合縮窄條件,就可以被轉換。
下面是C++11版本的校驗表,編譯器將按照下表依次執行。
1.如果有一位操作數的類型是long double,則另一個操作數轉換為long double。
2.否則,如果有一個操作數的類型是double,則另一個操作數轉換為double。
3.否則,如果有一個操作數的類型float,則另一個操作數轉換為float。
4.否則,說明操作數都是整形的,因此執行整形提升,什么是整形提升,下面有寫。
5.在整形提升的情況下,如果兩個操作數都是有符號或者無符號類型的,且其中一個操作數的級別比另一個低,則轉換為最高級別的類型。
6.如果一個操作數為有符號的,另一個操作數是無符號的,且無符號操作數的級別比有符號操作數的級別高,則將有符號操作數轉換為無符號操作數所屬的類型。
7.否則,如果有符號類型可以表示無符號類型的所有可能取值,則將無符號操作數轉換為有符號操作數所屬的類型。
8.否則,將兩個操作數都轉換為有符號類型的無符號版本。
整形提升:
如果bool,char、short,包括它們有符號或無符號變型,以及枚舉類型,可以使用在需要int或者unsigned int的表達式中。如果int可以完整表示源類型的所有值,那么該源類型的值就轉換為int,否則轉換為unsigned int。這稱為整型提升。
如果函數參數類型定義為double類型,但是傳入的時int類型,這在C中會提示錯誤,但在C++中,C++會自動幫我我們轉換為函數原型中定義的值,條件是兩種都是算術類型。也可以手動取消這種自動,在這種情況下,C++將對char和short類型進行整形提升,將float轉為double類型。
C++允許用戶自己強制轉換變量的類型,C++自己規定的類型轉換規則有時候可能并不適合用戶,并且被轉的變量本身并沒有有任何影響。
int a = 66; (long)a;// 這種是C的風格 long(a);// 這種是C++的風格,應盡量使用這種。
C++新增的一個工具,讓編譯器能夠根據初始值的類型推斷變量的類型,像是js中的var,這個東西就是C語言中的關鍵字auto。
auto a = 666;//編譯器將為a定義為int類型 auto b = 66.66f;//編譯器將為b定義為float類型,注意數值后面的f
以上就是C++中怎么實現類型轉換,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。