`
0769
  • 浏览: 83660 次
  • 性别: Icon_minigender_1
  • 来自: 东莞
社区版块
存档分类
最新评论

C++中的指针(一) 简单指针

阅读更多
C++中的指针(一) 简单指针
 

简单总结一下C++指针的用法,以后再写一篇详细的,关于smart pointer的总结。

指针的定义很简单。在变量前打个星。例如一个class的名字叫A,那么指针定义为
A  *pa;

有意点点另人混淆的是指针和const的混用。
char chr[] = "abc";
const char *p = chr; //这里p不是常数指针,而是把指针指向的地址定义为了常数。无论chr本身是不是指向常数内存区,但只要用p去操作,那么就不可以通过p去修改其内容。

chr[2] = 'e'; // ok
p[2] = 'd';  // error

p+=1; // ok, 改的是p指向的地址而不是p的内容。

真正的常数指针这么写
char *const cp = s;
这时在常数内存中allocate了一个指针的控件存储cp,cp,也就是这个地址不能改,而其指向的内存的值可以修改。
chr[2] = 'w';  //ok
cp[2] = 'y';  // ok
cp+=1;  // error

char* 可以被转换成const char*,因为操作后没有负面影响。反过来const char* 不能转换成char*,如果可以的话会把本部可写的内存的数据改掉。
// good example
char chr[] = "abc";
char *p = chr;
const char *cp = p;

// bad example
char chr[] = "abc";
const char *p = chr;
char *p = cp;  // error.

这种转换常用在函数调用上,例如strcpy(char* source, char*dest)。这个操作只是想修改source,dest只是用于参考。为了避免函数修改dest可以把函数定义成strcpy(char* source, const char* dest)。

基本定义就这些了。对于指针的cast,C++作得比C更安全。例如有两个完全不相干的class A和B。
   B b;
   A *p1,*p2;

p1 = (A *)(&b); // 这是C式的cast,不管A和B有什么关系,强型转换。后果不堪设想。
C++中引入了static_cast操作,在一定程度上保护了操作的安全性,static_cast检查操作数与要操作的类型是否匹配,匹配是有class继承关系,无论谁继承谁都可以。如果这个关系不存在,出编译错误。
   p2 = static_cast<A *>(&b);  //error。
但是这个检查是不完全的,
   class C : public A {}
   C* pc = static_cast<A *>(p1); // ok. 因为pc,p2欧继承关系。

C++引入了RTTI得概念(Run Time Type Info)。通过dynamic_cast操作,可以检查操作数的内容,以确认这个操作是否成功。检查内容的方法就是把相关类型的继承关系和vtable都查一下。
        p2 = dynamic_cast<A *>(p1);
在VC下使用dynamic_cast别忘了在当前Project-->Setting下选Enable Run Time Type Info。如果忘了选这个,debug模式下编译会不通过,release模式下会编译通过,运行时Crash。

dynamic_cast比较复杂,另外Visual C++各不同版本的表现不一样,这里详细说一下我学到的和试出来的。一般书上说是三种不同情况,考虑到Visual C++版本的问题,我分五个情况讨论。
1。upcast。从派生类向基类的转换,只要基类的继承关系是唯一的,就会成功,如果不唯一会有warning:"dynamic_cast used to convert to inaccessible or ambiguous base;"
下下面的例子中
class A{public:    virtual void a(){}};
class B : public A {};
class C : public B {};
class D : public B {};
class E : public C, public D {};

int main()
{
    E e, *pe = &e;
    C *pc = dynamic_cast<C*>(pe);
    B *pb = dynamic_cast<B*>(pe);
    return 0;
}
转换pb一行会有warning,而且得到NULL指针
其继承关系如下
                        A
                     /       \
                  B            B
                   |             |
                  C            D
                     \       /
                         E
E到C成功,E到B失败因为不知道怎么转换。同样E到A也会失败。注意这里的检查只是指针类型pe的检查,没有查pe指向的object。把pe改成*pe = (E*)(new D());的话pe到pc的cast还会成功,不过pe到pb的cast会出现crash。这和dynamic_cast的实现有关,这个exception不是bad_cast,所以最好用try{} chatch(...)接着以防不测。

2。对于同类指针的cast应该是直接通过, 不对指针所指的object进行run time check。
int main()
{
    A *p1 = (A*)0x1;
    A *p2 = dynamic_cast<A*>(p1);
    return 0;
}
但是VC6中竟对p1所指的地址进行了检查,这是VC6对ISO C++ standard实现不对的地方,在2003/2005中得到了修正。

3。downcast
从基类向派生类转换,指针指向的object会被检查,还以刚才的结构
    E e;
    A *pa = dynamic_cast<A*>((D*)&e);
    C *pc = dynamic_cast<C*>(pa);
这个pc的cast会成功。

4。crosscast
class A{public:    virtual void a(){}};
class B : public A {};
class C : public A {};
class D{public:       virtual void d(){}};
class E : public B, public C, public D {};

int main()
{
    E e;
    C *pc = dynamic_cast<C*>((D*)&e);
    return 0;
}
这个继承关系如下
     A
   /  |  \
B   C  D
   \  |  /
     E
从D到C的cast叫cross cast,这时查指针指的object的内容。这个具体例子中pc的cast 成功,因为确实有继承关系。
一个不理解的问题是下面的测试:
    C c, *pc = dynamic_cast<C*>((D*)&c);
无论什么道理pc都应该成功,结果在VC6,VC2003中都成功了,在VC2005竟然失败,得到NULL。实在不明白,在MSDN的"Breaking Changes in dynamic_cast"也没有明确表述。只有死记住了。

总之,dynamic_cast如果成功,p2得到一个合法地址,也就是p1指向的地址。如果失败就不好说了,书上说会得到NULL,这是理想情况,p1,p2有相近的vtable。如果p1,p2的vtable完全不相干,或者一个class B根本没有vtable,dynamic_cast就会出exception,这不是bad_cast的exception,而是C++ first class exception。所以写别人程序传来的指针的时候别指望dynamic_cast管理一切,老老实实catch所有exception。
    p2 = NULL;
    try
    {
        p2 = dynamic_cast<A *>(p1);
    }
    catch (...) {}

    if (!p2)
        cout << "Bad cast".


另外两种cast不太常用reinterpret_cast提供很少的保护,几乎和C的cast差不多。const_cast得到最开始的变量的指针,可以用来改变常量的设置。这不是个好习惯,能不用最好不用。

smart pointer C++中一个很有用的概念,它对内存的管理起到了很大的帮助。由于内容比较多,回头我会写一篇详细的总结。

分享到:
评论

相关推荐

    C++指针 数组 内存释放.docx

    C++指针是一种基本变量,包含一个实际的数据,该数据代表一个可以找到实际信息的内存地址。指针是C++编程中非常重要的概念,许多程序和思想依靠指针作为他们设计的基础。 定义指针变量 定义指针变量的方法与定义...

    c++中指针的详细介绍

    在C++编程语言中,指针是一个至关重要的概念,它为程序员提供了对内存直接操作的能力,使得高级编程技巧如动态内存分配、数据结构实现、函数指针等成为可能。本书《C++中指针的详细介绍》针对初学者,旨在帮助他们...

    C++中指针的使用艺术

    学习这些高级用法是提高C++指针操作技能的重要步骤。 指针的使用需要程序员有严格的编码习惯和清晰的逻辑思维。错误地使用指针可能会导致不可预知的错误,例如野指针(指向未知地址)、空悬指针(指向的内存已被...

    c++数组指针ppt

    指针是C++中的一种数据类型,它是一个变量,它存储了内存中某个位置的地址。指针的定义格式为:[&lt;存储种类&gt;] &lt;数据类型&gt; *&lt;标识符&gt; [= &lt;初值表达式&gt;]。例如:int *pa;这里,pa是一个整型指针,指向整型变量的地址。 ...

    C++双指针示例

    在C++编程中,双指针是一种非常常见且强大的技术,尤其在处理数组、链表以及其他数据结构时。双指针的基本思想是设置两个指针,一个从数组或数据结构的头部开始,另一个从尾部开始,然后逐渐靠近,直到找到特定条件...

    C/C++ 函数指针的意义,函数指针的用法

    在C/C++编程中,函数指针是一种高级特性,它允许程序员处理函数的方式如同处理变量一般灵活。理解函数指针的意义及其用法,对于提升代码的可扩展性和灵活性至关重要。 #### 1. 函数指针的基本概念 在C/C++中,函数...

    c++智能指针的创建

    在《Effective C++ 2/e》一书中提到,如果你的类里面有指针指向动态分配的内存,那么一定要为它写一个拷贝构造函数和一个赋值运算符。否则的话,一旦你做了赋值,会导致两个对象的指针指向同一块内存,对了!如果是...

    C++指针介绍.ppt

    C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针介绍.ppt C++指针...

    C/C++中多级指针的线性剖析

    在指针的使用中,最为简单的是直接指向基本数据类型的指针,即一级指针。当指针指向一个变量时,可以通过解引用运算符`*`来获取该变量的值。例如,如果`p`是一个指向整型变量`a`的指针,那么通过`*p`可以得到`a`的值...

    C++指针及句柄

    ### C++指针及句柄详解 #### 一、引言 在计算机编程领域,特别是对于像C++这样的强类型语言来说,理解指针是非常重要的。指针是一种特殊的数据类型,它存储的是其他数据的内存地址。掌握指针的使用不仅能够帮助...

    c++学习笔记(c++ 线程 指针 调试 编码)

    其次,指针是C++中非常核心且强大的概念:它是一个变量,存储的是其他变量的内存地址。指针可以用来动态分配内存、实现数据结构如链表和树,以及在函数之间传递复杂对象。理解指针的引用、解引用、空指针、野指针...

    c_c++指针详解 指针详解

    C/C++指针是编程语言中的一种基本数据类型,它们可以指向内存中的地址,可以存储变量的值,也可以存储函数的地址。指针的使用可以提高程序的效率和灵活性,但是也增加了程序的复杂度和出错的可能性。 ...

    C++指针——数组指针/函数指针

    在C++编程语言中,指针是至关重要的概念,它允许我们存储内存地址并直接操纵内存。本篇文章将深入探讨两种特殊的指针类型:数组指针和函数指针。这两种类型的指针在处理数据集合和调用函数时具有显著的优势。 首先...

    C++指针与引用的区别

    C++指针与引用的区别 指针和引用是C++语言中两种不同的概念,初学者容易把它们混淆一起。这两种概念都是地址的概念,但它们有很多不同的特点和应用场景。 相同点 1. 都是地址的概念;指针指向一块内存,它的内容...

    C++指针精髓 笔记

    本笔记将深入探讨C++指针的精髓,包括其基本概念、使用方法以及高级应用。 首先,我们要理解指针的基本原理。在C++中,指针是一个变量,它存储的是另一个变量的内存地址。当我们声明一个指针变量时,例如`int *p;`...

    C++中函数指针的含义

    在C++中,函数指针是一个非常重要的概念,它允许我们直接操作函数的地址,从而实现函数的动态调用和传递。理解函数指针对于深入掌握C++编程至关重要。 首先,函数在内存中占据一定的位置,就像数据一样,它们也有...

    C++智能指针测试代码

    C++ 7种智能指针测试代码

    C++ 指针 最好例子

    本文将通过五个实例深入理解C++指针的使用。 实验一介绍了指针的基本概念,如何创建指向普通变量的指针并进行赋值。在C++中,可以使用`int *p1`声明一个指向整型变量的指针,`&k`获取变量k的地址并赋值给`p1`。`*p1...

Global site tag (gtag.js) - Google Analytics