C++语言中数组指针和指针数组彻底分析

################################
#                              #
#       基本知识               #
#                              #
################################

当然我们一切都是从最简单的内建类型开始,最后我会做一些推广。
先看一下基本的形式,我们从这里起步!

Cpp代码

  1. --------------指针----------------
  2. int a=10;
  3. int *p=&a;
  4. -------------指针的指针-----------
  5. int b=20;
  6. int *p=&b;
  7. int **p2p=&p;
  8. -------------简单数组-----------------
  9. int c[10];//整数数组,含有10个整数元素
  10. file://也就是说每一个元素都是整数
  11. --------------指针数组--------------------
  12. int *p[10];//指针数组,含有10个指针元素
  13. file://也就是说每一个元素都是指针
  14. --------------数组指针--------------------
  15. int (*p)[10];//数组指针,这个指针可以用来指向
  16. file://含有10个元素的整数数组

上面这些简单的形式是我们必须要首先理解,这个是基本的知识。
同时我们从上面也要得出一个很重要的知识提示:c++语言层面上
关于变量声明的部分,后缀结合变量的优先级比前缀要高的。
看我们上面的例子的最后两个就明白了,我们为了实现数组指针的
声明我们不得不变通一下。我们采用()来实现优先级的改变,实现了
数组指针的声明。

################################
#                              #
#      进一步提高知识          #
#                              #
################################

数组,数组的指针,指针的数组,概念太多了。我接受概念一多的
时候,我就想把这些复杂的东西简单一下。因为我太懒了,概念简化
一下,记住更容易一点。所以我们这里要认识一下上面这些概念本质。
这样可以简化概念,减少记忆的难度。

先看一段程序。

Cpp代码

  1. #include <iostream>
  2. #include <typeinfo>
  3. using namespace std;
  4. int main()
  5. {
  6. int vInt=10;
  7. int arr[2]={10,20};
  8. int *p=&vInt;
  9. int **p2p=&p;
  10. int *parr[2]={&vInt,&vInt};
  11. int (*p2arr)[2]=&arr;
  12. cout<<"Declaration [int vInt=10] type=="<<typeid(vInt).name()<<endl;
  13. cout<<"Declaration [arr[2]={10,20}] type=="<<typeid(arr).name()<<endl;
  14. cout<<"Declaration [int *p=&vInt] type=="<<typeid(p).name()<<endl;
  15. cout<<"Declaration [int **p2p=&p] type=="<<typeid(p2p).name()<<endl;
  16. cout<<"Declaration [int *parr[2]={&vInt,&vInt}] type=="<<typeid(parr).name()<<endl;
  17. cout<<"Declaration [int (*p2arr)[2]=&arr] type=="<<typeid(p2arr).name()<<endl;
  18. return 0;
  19. }

运行的结果如下:(我在前面加了行号#XX)
#01 Declaration [int vInt=10] type==int
#02 Declaration [arr[2]={10,20}] type==int *
#03 Declaration [int *p=&vInt] type==int *
#04 Declaration [int **p2p=&p] type==int * *
#05 Declaration [int *parr[2]={&vInt,&vInt}] type==int **
#06 Declaration [int (*p2arr)[2]=&arr] type==int (*)[2]

现在我们来分析一下结果。因为我们已经具有了第一部分的基本知识,我们现在
可以很明确区别出来我们声明的类型。这里主要有两个很重要的部分,我们不过
是就事讲事情,编译器是如何实现的原理不在这里讨论之列。

--------#02:数组------------

现在看#02,想到了什么没有呀?在编译器看来数组只是相对应类型的指针类型。
当我们把数组传递给函数作为参数的时候,传递的是指针,所以我们可以利用
参数来修改数组元素。这个转化是编译器自动完成的。

void f(int[]);
int a[2]={10,20};
f(a);//这行等价于编译器完成的函数转化f(int *p)

也就是说这里编译器自动完成了int[]类型到int *的转化,
注意是编译器完成的,也可以说是语言本身实现的,我们
对此只有接受的份了。

-------#05:指针数组---------------

指针数组的编译器内部表示也是对应类型的指针。

------#06:数组指针----------------
数组指针的编译器内部表示就是有一点特别了。
编译器(或者说是语言本身)有数组指针这个内部表示。
由于c++语言的类型严格检查的语言(当然还有一些是存在隐式类型转化的)

所以我们下面的写法是不能编译通过的。
{
file://---------编 译不能通过--------------
int arr[3]={10,20};//注意是3个元素数组
int (*p2arr)[2]=&arr;//注意是指向2个元素数组的指针
file://---------编 译不能通过--------------
}

################################
#                              #
#      初步小结                #
#                              #
################################

通过上面两个小节的内容,大家应该基本明白了,
数组,指针,指针数组,数组指针到底是怎么一回事情了吧。

-----------补充开始-----------------------
关于数组和指针的转化,以及我们使用指针(++,--)等来操作数组,
是基于数组在内存中是连续分布的。

但是我们使用“迭代器”的时候,情况是不一样的。
这个问题本文不讨论。

-----------补充结束---------------------

不过c++语言本身有很多诡异的地方(因为c++要考虑到跟c语言以及旧的c++版本兼容)。
内建类型的这些性质特征到了函数部分会有一点小的变化,不过如果你了解了编译器做了
什么以后的话,你也就不会太奇怪了。不过关于函数部分的内容我下次再说了。

现在回到上面的内容。我们这里还是讲一下内建类型。显然一样类型的变量是可以互相赋值。
不过当然还有一些其他情况也是可以的,比如类型的宽化,关于类的继承体系问题等等。

当然了,不一样的类型一般来说是不能互相赋值,当然这里的例外就是强制转化,
类的继承体系等情况了。

看到这里就会明白下面的程序为什么会运行的了。
我这里也把下面的程序作为今天内容的总结:

Cpp代码

  1. #include <iostream>
  2. using namespace std;
  3. int main()
  4. {
  5. int a[2]={10,20};
  6. int *p=a;//根据上面说明,由于编译器的参与,两者类型转化后一致
  7. int vInt=10;
  8. int *parr[2]={&vInt,&vInt};
  9. int **p2p=parr;//上面分析,类型一致
  10. return 0;
  11. }
时间: 2024-10-18 01:16:54

C++语言中数组指针和指针数组彻底分析的相关文章

(转)C语言中长度为0的数组

前面在看Xen的源码时,遇到了一段代码,如下所示: 注意上面最后一行的代码,这里定义了一个长度为的数组,这种用法可以吗?为什么可以使用长度为0 的数组?长度为的数组到底怎么使用?……这篇文章主要针对该问题进行简单的讲解.废话不多说了,现在就开始. 长度为的数组在标准c和c++中是不允许的,如果使用长度为的数组,编译时会产生错误,提示数组长度不能为.但在GNUc中,这种用法却是合法的.它的最典型的用法就是位于数组中的最后一项,如上面所示,这样做主要是为了方便内存缓冲区的管理.如果你将上面的长度为的

以杨辉三角为例,从内存角度简单分析C语言中的动态二维数组

学C语言,一定绕不过指针这一大难关,而指针最让人头疼的就是各种指向关系,一阶的指针还比较容易掌握,但一旦阶数一高,就很容易理不清楚其中的指向关系,现在我将通过杨辉三角为例,我会用四种方法从内存的角度简单分析动态二维数组,若有不足或错误之处,还请指出! 在讲这之前,以一维数组为例,先重新认识一下数组: int array[5] = {1, 2, 3, 4, 5}; 首先数组名称是该数组的首地址常量,即数组名称就是指针,就有&array[0] == array! 那么我们可以推出*array ==

C语言中 指向函数的指针 简介

引子:在学习CPrimerPlus的第十四章的14.13节中,遇到了如下三行文字,是有关指向函数的指针的,把我搞晕了. char * fump(); //返回指向char的指针的函数 char (* frump)(); //指向返回类型为char的函数的指针 char (* flump[3])(); //由3个指针组成的数组,每个指针指向返回类型为char的函数 原来自己根本都不知道还有指向指针的函数,难怪自己懵逼了.参考了两篇博客(1."http://blog.csdn.net/hzyong_

c语言中的结构体指针类型的cast

1.我们在c语言中会经常碰到强制类型转换. 在这,我介绍一种结构pointer类型转换,但是有前提(有点类似于c++中的继承中的子父对象的cast). 简单的介绍一下: 首先我们要知道一个结构的指针,并且 在这个结构体中,第一个结构成员必须也是一个结构体(最好是结构体类型). 那么我们可以这个结构体指针转换为指向这个结构体中第一个成员结构体的指针. 直接看代码: ************************************ /* struct transform for struct

Go语言中地址传递及指针的理解

在学习指针时遇到一个问题,在网上贴出后,得到很多前辈的指导和耐心解答,在此表示感谢,因此整理了此篇文章. 第一次实验,代码如下: type Student struct { Name string } func remove(stu *Student) { fmt.Printf("2----%v\n", stu) stu = nil fmt.Printf("3----%v\n", stu) } func main() { stu := &Student{&q

c语言中指向整型指针的指针的理解

1 /************************************************************************* 2 > File Name: ptr_ptr_int.c 3 > Author: Mr.Yang 4 > Purpose:演示指向整型的指针的指针 5 > Created Time: 2017年06月03日 星期六 18时34分58秒 6 **********************************************

c语言中函数调用的本质从汇编角度分析

今天下午写篇博客吧,分析分析c语言中函数调用的本质,首先我们知道c语言中函数的本质就是一段代码,但是给这段代码起了一个名字,这个名字就是他的的这段代码的开始地址 这也是函数名的本质,其实也就是汇编中的标号.下面我们会接触到一些东西 比如 eip 就是我们常常说的程序计数器,还有ebp和esp (这里是俩个指针,记得我们以前学8086也就一个sp堆栈指针)分别为EBP是指向栈底的指针,在过程调用中不变,又称为帧指针.ESP指向栈顶,程序执行时移动,ESP减小分配空间,ESP增大释放空间,ESP又称

c语言中如何通过二级指针来操作二维数组

通过二级指针去访问二维数组需要先给二级指针分配等同于二维数组行数的一维数组指针,然后把二维数组的每行首地址赋值给对应位置的一维指针上.之后就可以通过二维指针直接访问了. 参考代码如下,可以看具体注释辅助理解. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 #include <stdio.h>//输入输出头文件. #include <stdlib.h>//本程序需要用到malloc/free函数,引

D语言中无法捕获函数指针为null的异常

import std.stdio; int main(string[] argv) { try { auto f = delegate(){}; f = null; f(); } catch(Throwable e) { writeln(e.msg); } readln(); return 0; } 在调试中,函数或委托的指针为null时,异常无法捕获,直接退出.只有在程序运行时才能捕获得到. 以上问题需要打开VS异常设置:

c/c++ 函数指针 指针函数 数组的引用 指针数组 数组指针

1.指针数组数组指针 引用数组 数组的引用 int *a[10] 指针数组 每一个元素都是一个指针 Int (*a)[10] 数组指针 P指向一个含有10个元素的数组 Int (&a)[10] 数组的引用 a是一个数组的引用 Int& a[10] 引用函数 非法 数组的引用:1.在程序体中 int a[10]; Int (&p)[10]=a;//引用数组 2.作为参数 #include <iostream> #include <string> using n