溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C++11中l(wèi)onglong超長整型和nullptr初始化空指針怎么使用

發(fā)布時間:2023-01-03 10:26:40 來源:億速云 閱讀:83 作者:iii 欄目:開發(fā)技術(shù)

這篇“C++11中l(wèi)onglong超長整型和nullptr初始化空指針怎么使用”文章的知識點大部分人都不太理解,所以小編給大家總結(jié)了以下內(nèi)容,內(nèi)容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“C++11中l(wèi)onglong超長整型和nullptr初始化空指針怎么使用”文章吧。

1. C++11:long long 超長整型

C++ 11 標準中,基于整數(shù)大小的考慮,共提供了如下表所示的這些數(shù)據(jù)類型。與此同時,標準中還明確限定了各個數(shù)據(jù)類型最少占用的位數(shù)。

整數(shù)類型等價類型C++11標準規(guī)定占用最少位數(shù)
shortshort int(有符號短整型)至少 16 位(2 個字節(jié))
signed shortshort int(有符號短整型)至少 16 位(2 個字節(jié))
signed short intshort int(有符號短整型)至少 16 位(2 個字節(jié))
unsigned shortunsigned short int(無符號短整型)至少 16 位(2 個字節(jié))
unsigned short intunsigned short int(無符號短整型)至少 16 位(2 個字節(jié))
intint(有符號整形)至少 16 位(2 個字節(jié))
signedint(有符號整形)至少 16 位(2 個字節(jié))
signed intint(有符號整形)至少 16 位(2 個字節(jié))
unsignedunsigned int(無符號整形)至少 16 位(2 個字節(jié))
unsigned intunsigned int(無符號整形)至少 16 位(2 個字節(jié))
longlong int(有符號長整形)至少 32 位(4 個字節(jié))
long intlong int(有符號長整形)至少 32 位(4 個字節(jié))
signed longlong int(有符號長整形)至少 32 位(4 個字節(jié))
signed long intlong int(有符號長整形)至少 32 位(4 個字節(jié))
unsigned longunsigned long int(無符號長整形)至少 32 位(4 個字節(jié))
unsigned long intunsigned long int(無符號長整形)至少 32 位(4 個字節(jié))
long long(C++11)long long int(有符號超長整形)至少 64 位(8 個字節(jié))
long long int(C++11)long long int(有符號超長整形)至少 64 位(8 個字節(jié))
signed long long(C++11)long long int(有符號超長整形)至少 64 位(8 個字節(jié))
signed long long int(C++11)long long int(有符號超長整形)至少 64 位(8 個字節(jié))
unsigned long long(C++11)unsigned long long int(無符號超長整型)至少 64 位(8 個字節(jié))
unsigned long long int(C++11)unsigned long long int(無符號超長整型)至少 64 位(8 個字節(jié))

C++11 標準規(guī)定,每種整數(shù)類型必須同時具備有符號(signed)和無符號(unsigned)兩種類型,且每種具體的有符號整形和無符號整形所占用的存儲空間(也就是位數(shù))必須相同。不過需要注意的是,C++11 標準中只限定了每種類型最少占用多少存儲空間,不同的平臺可以占用不同的存儲空間。

在上表羅列的這些數(shù)據(jù)類型中,long long 超長整型是 C++ 11 標準新添加的。其實早在 1995 年,就有人提議將 long long 整形寫入 C++ 98 標準,但被委員會拒絕了。而后 long long 整形被 C99 標準(C語言標準之一)采納,并逐漸被很多編譯器支持,于是 C++ 標準委員會重新決定將 long long 整形寫入 C++ 11 標準中。

如同 long 類型整數(shù)需明確標注 "L" 或者 "l" 后綴一樣,要使用 long long 類型的整數(shù),也必須標注對應的后綴:

  • 對于有符號 long long 整形,后綴用 "LL" 或者 "ll" 標識。例如,"10LL" 就表示有符號超長整數(shù) 10;

  • 對于無符號 long long 整形,后綴用 "ULL"、"ull"、"Ull" 或者 "uLL" 標識。例如,"10ULL" 就表示無符號超長整數(shù) 10。

如果不添加任何標識,則所有的整數(shù)都會默認為 int 類型。

對于 long long 類型來說,如果想了解當前平臺上 long long 整形的取值范圍,可以使用<climits>頭文件中與 long long 整形相關(guān)的 3 個宏,分別為 LLONG_MIN、LLONG_MAX 和 ULLONG_MIN:
1)LLONG_MIN:代表當前平臺上最小的 long long 類型整數(shù);
2)LLONG_MAX:代表當前平臺上最大的 long long 類型整數(shù);
3)ULLONG_MIN:代表當前平臺上最大的 unsigned long long 類型整數(shù)(無符號超長整型的最小值為 0)。
舉個例子:

#include <iostream>
#include <iomanip>
#include <climits>
using namespace std;

int main()
{
    cout <<"long long最大值:" << LLONG_MIN <<" "<< hex << LLONG_MIN <<"\n";
    cout << dec <<"long long最小值:" << LLONG_MAX << " " << hex << LLONG_MAX << "\n";
    cout << dec << "unsigned long long最大值:" << ULLONG_MAX << " " << hex << ULLONG_MAX;
    return 0;
}

程序執(zhí)行結(jié)果為(不唯一):

long long最大值:-9223372036854775808 8000000000000000
long long最小值:9223372036854775807 7fffffffffffffff
unsigned long long最大值:18446744073709551615 ffffffffffffffff

此程序中,輸出了各最大值和最小值對應的十六進制,顯然在當前平臺(Windows10 64位操作系統(tǒng))上,long long 超長整型占用 64 位(也就是 16 個字節(jié))的存儲空間。

2. C++11:nullptr 初始化空指針

實際開發(fā)中,避免產(chǎn)生“野指針”最有效的方法,就是在定義指針的同時完成初始化操作,即便該指針的指向尚未明確,也要將其初始化為空指針。

所謂“野指針”,又稱“懸掛指針”,指的是沒有明確指向的指針。野指針往往指向的是那些不可用的內(nèi)存區(qū)域,這就意味著像操作普通指針那樣使用野指針(例如 &p),極可能導致程序發(fā)生異常。

C++98/03 標準中,將一個指針初始化為空指針的方式有 2 種:

int *p = 0;
int *p = NULL; //推薦使用

可以看到,我們可以將指針明確指向 0(0x0000 0000)這個內(nèi)存空間。一方面,明確指針的指向可以避免其成為野指針;另一方面,大多數(shù)操作系統(tǒng)都不允許用戶對地址為 0 的內(nèi)存空間執(zhí)行寫操作,若用戶在程序中嘗試修改其內(nèi)容,則程序運行會直接報錯。
相比第一種方式,我們更習慣將指針初始化為 NULL。值得一提的是,NULL 并不是 C++ 的關(guān)鍵字,它是 C++ 為我們事先定義好的一個宏,并且它的值往往就是字面量 0(#define NULL 0)。

C++ 中將 NULL 定義為字面常量 0,雖然能滿足大部分場景的需要,但個別情況下,它會導致程序的運行和我們的預期不符。例如:

#include <iostream>
using namespace std;

void isnull(void *c){
    cout << "void*c" << endl;
}
void isnull(int n){
    cout << "int n" << endl;
}

int main() {
    isnull(0);
    isnull(NULL);
    return 0;
}

程序執(zhí)行結(jié)果為:

int n
int n

對于 isnull(0) 來說,顯然它真正調(diào)用的是參數(shù)為整形的 isnull() 函數(shù);而對于 isnull(NULL),我們期望它實際調(diào)用的是參數(shù)為 void*c 的 isnull() 函數(shù),但觀察程序的執(zhí)行結(jié)果不難看出,并不符合我們的預期。
C++ 98/03 標準中,如果我們想令 isnull(NULL) 實際調(diào)用的是 isnull(void* c),就需要對 NULL(或者 0)進行強制類型轉(zhuǎn)換:

isnull( (void*)NULL );
isnull( (void*)0 );

如此,才會成功調(diào)用我們預期的函數(shù)。

由于 C++ 98 標準使用期間,NULL 已經(jīng)得到了廣泛的應用,出于兼容性的考慮,C++11 標準并沒有對 NULL 的宏定義做任何修改。為了修正 C++ 存在的這一 BUG,C++ 標準委員會最終決定另其爐灶,在 C++11 標準中引入一個新關(guān)鍵字,即 nullptr。

在使用 nullptr 之前,需保證自己使用的編譯器支持該關(guān)鍵字。以 Visual Studio 和 codeblocks 為例,前者早在 2010 版本就對 C++ 11 標準中的部分特性提供了支持,其中就包括 nullptr;如果使用后者,則需將其 G++ 編譯器版本至少升級至 4.6.1(同時開啟 -std=c++0x 編譯選項)。

nullptr 是 nullptr_t 類型的右值常量,專用于初始化空類型指針。nullptr_t 是 C++11 新增加的數(shù)據(jù)類型,可稱為“指針空值類型”。也就是說,nullpter 僅是該類型的一個實例對象(已經(jīng)定義好,可以直接使用),如果需要我們完全定義出多個同 nullptr 完全一樣的實例對象。

值得一提的是,nullptr 可以被隱式轉(zhuǎn)換成任意的指針類型。舉個例子:

int * a1 = nullptr;
char * a2 = nullptr;
double * a3 = nullptr;

顯然,不同類型的指針變量都可以使用 nullptr 來初始化,編譯器分別將 nullptr 隱式轉(zhuǎn)換成 int*、char* 以及 double* 指針類型。

另外,通過將指針初始化為 nullptr,可以很好地解決 NULL 遺留的問題,比如:

#include <iostream>
using namespace std;

void isnull(void *c){
    cout << "void*c" << endl;
}
void isnull(int n){
    cout << "int n" << endl;
}

int main() {
    isnull(NULL);
    isnull(nullptr);
    return 0;
}

程序執(zhí)行結(jié)果為:

int n
void*c

借助執(zhí)行結(jié)果不難看出,由于 nullptr 無法隱式轉(zhuǎn)換為整形,而可以隱式匹配指針類型,因此執(zhí)行結(jié)果和我們的預期相符。

總之在 C++11 標準下,相比 NULL 和 0,使用 nullptr 初始化空指針可以令我們編寫的程序更加健壯。

以上就是關(guān)于“C++11中l(wèi)onglong超長整型和nullptr初始化空指針怎么使用”這篇文章的內(nèi)容,相信大家都有了一定的了解,希望小編分享的內(nèi)容對大家有幫助,若想了解更多相關(guān)的知識內(nèi)容,請關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI