(转)null和NULL和nullptr和””区别

突然想到这个有趣的问题:C语言和C++对大小写是敏感的,也就是说null和NULL是区别对待的。NULL代表空地址,null只是一个符号。便来深究,看了很多资料,总结如下:

 

其实null和NULL都是字符串(没啥区别,欢迎高手纠错!),具体看它们宏定义被定义成为什么值。

在VS中NULL被定义为0,因为习惯上把宏定义的所有字符都大写,当把NULL它赋值给指针时意思为空,当然你也可以把null定义为0了,都一样。

它们都没定义时都只能叫符号了,定义后就有另外的意思了,你把0直接赋值给指针也行,只要指针指向0就为空。

其实NULL在有些编译器中是赋为0了,这时你不能再去#define定义它了,否则就重定义了!

 

1,NULL

NULL的定义

[cpp]  

#ifdef __cplusplus  

#define NULL    0  

#else  

#define NULL    ((void *)0) 

#endif  

在C语言环境下,由于不存在函数重载等问题,直接将NULL定义为一个void*的指针就可以完美的解决一切问题。

但是在C++环境下情况就变得复杂起来,首先我们不能写这样的代码:

FILE* fp = (void*)0; //将void*直接赋值给一个指针是不合法的,编译器会报错。

我们只能这样写代码

[cpp]  

FILE* fp = (FILE*)0;

// or  

FILE* fp = 0;

 

/*

所以在C++中,NULL就被直接定义为一个整型0。在大多数情况下这并不会产生什么问题。但是万一有重载或者模板推导的时候,编译器就无法给出正确结果了。比如下面的情形:

[cpp] 

voidcall_back_process(CCObject* target, void* data);  

bind(call_back_process,target, NULL); // error 函数类型是void* ,但是我们绑定的是一个整型 0。

 */

2,nullptr

C++11,其中有一个是新的关键字nullptr

如果我们的编译器是支持nullptr的话,那么我们应该直接使用nullptr来替代NULL的宏定义。正常使用过程中他们是完全等价的。

 

模拟nullptr的实现:

某些编译器不支持c++11的新关键字nullptr,我们也可以模拟实现一个nullptr

[cpp]

const  

class nullptr_t_t  

{  

public:  

    template<class T>          operator T*() const {return 0;}  

    template<class C, classT>  operator T C::*() const { return 0; }  

private:  

    void operator& () const; 

} nullptr_t = {};  

#undef NULL  

#define NULL nullptr_t  

 

3,“”

“”也表示空,但是和null有很大区别:null没有分配空间,""分配了空间

String str1 = null;  //str引用为空

String str2 = "";  //str引用一个空串

也就是null没有分配空间,""分配了空间,因此str1还不是一个实例化的对象,而str2已经实例化。


注意:

因为null不是对象,""是对象。所以比较的时候必须是 if(str1==null){...}和if(str2.equals("")){...}。

对象用equals比较,null用等号比较。

因此,如果str1=null;下面的写法错误:

if(str1.equals("")||str1==null){//如果str1没有值,则.... } 

正确的写法是

if(str1==null||str1.equals("")){//先判断是不是对象,如果是,再判断是不是空字符串}

打个比方:一个空玻璃杯,你不能说它里面什么都没有,因为里面有空气,当然也可以把它弄成真空,null与" "的区别就象真空与空气一样。

 

Thanks.

源博客地址:http://blog.csdn.net/cc1949/article/details/51249555

posted @ 2017-02-06 10:57  公子綦  阅读(3415)  评论(0编辑  收藏  举报