一、NULL和nullptr區別
- NULL是宏定義,nullptr是關鍵字。
#ifndef NULL#ifdef __cplusplus#define NULL 0#else#define NULL ((void *)0)#endif
#endif
- nullptr可以隱式轉換為任意指針類型,而NULL需要顯示轉換
void func(char *)
{std::cout << "char * func" << std::endl;
}
void func(int)
{std::cout << "int func" << std::endl;
}int main()
{//NULL; nullptr;func(NULL); //"int func"func((char *)NULL); //"char * func"func(nullptr); //"char * func"}
- NULL可以進行算數運算,nullptr不行
std::cout << NULL + 1 << std::endl; //1//std::cout << nullptr + 1 << std::endl; //編譯器報錯
- 比較運算也有區別,nullptr可以與任意指針或nullptr_t類型以及0比較(非0整數不允許),而NULL可以與整數和指針比較。
int *a = new int(4);if (nullptr < a)std::cout << "nullptr < a" << std::endl;if (NULL < a)std::cout << "NULL < a" << std::endl;
二、NULL和nullptr相同點
- 兩者都不能取地址,但是nullptr_t類型的變量可以取地址。
- 兩者的大小相等,且與平臺相關。
- 兩者都能與指針和0的比較。
std::cout << sizeof(NULL)<< sizeof(nullptr)<<"\n"<<(0==NULL) << (0==nullptr)<<std::endl; //8,8,true,true
if (nullptr==0)std::cout << "nullptr\n"<<std::endl; //nullptr
三、nullptr與nullptr_t的特性
- nullptr_t類型是由nullptr獲取的
typedef decltype(nullptr) nullptr_t;
- 所有nullptr_t類型的變量行為都與nullptr等價,行為也一樣
- 都能隱式轉換為任意指針
- 不能轉為非指針類型,即使reinterpret_cast<nullptr_t>()也不行。(實際上可以轉為bool)
- 不能進行算術運算,
- 能進行關系運算,支持nullptr_t類型和指針類型,以及0,其他整數都不行
- 是類型,所以模板時不支持推導成T *
using namespace std;template<typename T> void g(T* t) {}template<typename T> void h(T t) {}int main(){g(nullptr); // 編譯失敗, nullptr的類型是nullptr_t,而不是指針g((float*) nullptr); // 推導出T = floath(0); // 推導出T = inth(nullptr); // 推導出T = nullptr_th((float*)nullptr); // 推導出T = float*}// 編譯選項:g++ 7-1-4.cpp -std=c++11