设为首页 加入收藏

TOP

2.3.2 指针(2)
2013-10-07 16:26:44 来源: 作者: 【 】 浏览:92
Tags:2.3.2 指针

2.3.2  指针(2)

空指针

空指针(null pointer)不指向任何对象,在试图使用一个指针之前代码可以首先检查它是否为空。以下列出几个生成空指针的方法:
 

  1. int *p1 = nullptr;      // 等价于int *p1 = 0;  
  2. int *p2 = 0;            // 直接将p2初始化为字面常量0  
  3. // 需要首先#include cstdlib  
  4. int *p3 = NULL;             // 等价于int *p3 = 0

得到空指针最直接的办法就是用字面值nullptr来初始化指针,这也是C++(www.cppentry.com)11新标准刚刚引入的一种方法。nullptr是一种特殊类型的字面值,它可以被转换成(参见2.1.2节,第35页)任意其他的指针类型。另一种办法就如对p2的定义一样,也可以通过将指针初始化为字面值0来生成空指针。

过去的程序还会用到一个名为NULL的预处理变量(preprocessor variable)来给指针赋值,这个变量在头文件cstdlib中定义,它的值就是0。

2.6.3节(第77页)将稍微介绍一点关于预处理器的知识,现在只要知道预处理器是运行于编译过程之前的一段程序就可以了。预处理变量不属于命名空间std,它由预处理器负责管理,因此我们可以直接使用预处理变量而无须在前面加上std::。

当用到一个预处理变量时,预处理器会自动地将它替换为实际值,因此用NULL初始化指针和用0初始化指针是一样的。在新标准下,现在的C++(www.cppentry.com)程序最好使用nullptr,同时尽量避免使用NULL。

把int变量直接赋给指针是错误的操作,即使int变量的值恰好等于0也不行。

  1. int zero = 0;  
  2. pi = zero;      // 错误:不能把int变量直接赋给指针 

建议:初始化所有指针

使用未经初始化的指针是引发运行时错误的一大原因。

和其他变量一样,访问未经初始化的指针所引发的后果也是无法预计的。通常这一行为将造成程序崩溃,而且一旦崩溃,要想定位到出错位置将是特别棘手的问题。

在大多数编译器环境下,如果使用了未经初始化的指针,则该指针所占内存空间的当前内容将被看作一个地址值。访问该指针,相当于去访问一个本不存在的位置上的本不存在的对象。糟糕的是,如果指针所占内存空间中恰好有内容,而这些内容又被当作了某个地址,我们就很难分清它到底是合法的还是非法的了。

因此建议初始化所有的指针,并且在可能的情况下,尽量等定义了对象之后再定义指向它的指针。如果实在不清楚指针应该指向何处,就把它初始化为nullptr或者0,这样程序就能检测并知道它没有指向任何具体的对象了。

赋值和指针

指针和引用都能提供对其他对象的间接访问,然而在具体实现细节上二者有很大不同,其中最重要的一点就是引用本身并非一个对象。一旦定义了引用,就无法令其再绑定到另外的对象,之后每次使用这个引用都是访问它最初绑定的那个对象。

指针和它存放的地址之间就没有这种限制了。和其他任何变量(只要不是引用)一样,给指针赋值就是令它存放一个新的地址,从而指向一个新的对象:
 

  1. int i = 42;  
  2. int *pi = 0;        // pi被初始化,但没有指向任何对象  
  3. int *pi2 = &i;      // pi2被初始化,存有i的地址  
  4. int *pi3;           // 如果pi3定义于块内,则pi3的值是无法确定的  
  5. pi3 = pi2;          // pi3和pi2指向同一个对象i  
  6. pi2 = 0;            // 现在pi2不指向任何对象了 

有时候要想搞清楚一条赋值语句到底是改变了指针的值还是改变了指针所指对象的值不太容易,最好的办法就是记住赋值永远改变的是等号左侧的对象。当写出如下语句时,
 

  1. pi = &ival;             // pi的值被改变,现在pi指向了ival 

意思是为pi赋一个新的值,也就是改变了那个存放在pi内的地址值。相反的,如果写出如下语句,

  1. *pi = 0;            // ival的值被改变,指针pi并没有改变 

则*pi(也就是指针pi指向的那个对象)发生改变。

其他指针操作

只要指针拥有一个合法值,就能将它用在条件表达式中。和采用算术值作为条件(参见2.1.2节,第35页)遵循的规则类似,如果指针的值是0,条件取false:
 

  1. int ival = 1024;  
  2. int *pi = 0;            // pi合法,是一个空指针  
  3. int *pi2 = &ival;       // pi2是一个合法的指针,存放着ival的地址  
  4. if (pi)                     // pi的值是0,因此条件的值是false  
  5.     // ...  
  6. if (pi2)                // pi2指向ival,因此它的值不是0,条件的值是true  
  7.     // ... 

】【打印繁体】【投稿】【收藏】 【推荐】【举报】【评论】 【关闭】 【返回顶部
分享到: 
上一篇2.3.2 指针(1) 下一篇2.3.2 指针(3)

评论

帐  号: 密码: (新用户注册)
验 证 码:
表  情:
内  容:

·C++ Lambda表达式保 (2025-12-26 05:49:45)
·C++ Lambda表达式的 (2025-12-26 05:49:42)
·深入浅出 C++ Lambda (2025-12-26 05:49:40)
·C语言指针从入门到基 (2025-12-26 05:21:36)
·【C语言指针初阶】C (2025-12-26 05:21:33)