C++语言中数组指针和指针数组彻底分析
################################
# #
# 基本知识 #
# #
################################
当然我们一切都是从最简单的内建类型开始,最后我会做一些推广。
先看一下基本的形式,我们从这里起步!
- --------------指针----------------
- int a=10;
- int *p=&a;
- -------------指针的指针-----------
- int b=20;
- int *p=&b;
- int **p2p=&p;
- -------------简单数组-----------------
- int c[10];//整数数组,含有10个整数元素
- file://也就是说每一个元素都是整数
- --------------指针数组--------------------
- int *p[10];//指针数组,含有10个指针元素
- file://也就是说每一个元素都是指针
- --------------数组指针--------------------
- int (*p)[10];//数组指针,这个指针可以用来指向
- file://含有10个元素的整数数组
上面这些简单的形式是我们必须要首先理解,这个是基本的知识。
同时我们从上面也要得出一个很重要的知识提示:c++语言层面上
关于变量声明的部分,后缀结合变量的优先级比前缀要高的。
看我们上面的例子的最后两个就明白了,我们为了实现数组指针的
声明我们不得不变通一下。我们采用()来实现优先级的改变,实现了
数组指针的声明。
################################
# #
# 进一步提高知识 #
# #
################################
数组,数组的指针,指针的数组,概念太多了。我接受概念一多的
时候,我就想把这些复杂的东西简单一下。因为我太懒了,概念简化
一下,记住更容易一点。所以我们这里要认识一下上面这些概念本质。
这样可以简化概念,减少记忆的难度。
先看一段程序。
- #include <iostream>
- #include <typeinfo>
- using namespace std;
- int main()
- {
- int vInt=10;
- int arr[2]={10,20};
- int *p=&vInt;
- int **p2p=&p;
- int *parr[2]={&vInt,&vInt};
- int (*p2arr)[2]=&arr;
- cout<<"Declaration [int vInt=10] type=="<<typeid(vInt).name()<<endl;
- cout<<"Declaration [arr[2]={10,20}] type=="<<typeid(arr).name()<<endl;
- cout<<"Declaration [int *p=&vInt] type=="<<typeid(p).name()<<endl;
- cout<<"Declaration [int **p2p=&p] type=="<<typeid(p2p).name()<<endl;
- cout<<"Declaration [int *parr[2]={&vInt,&vInt}] type=="<<typeid(parr).name()<<endl;
- cout<<"Declaration [int (*p2arr)[2]=&arr] type=="<<typeid(p2arr).name()<<endl;
- return 0;
- }
运行的结果如下:(我在前面加了行号#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++版本兼容)。
内建类型的这些性质特征到了函数部分会有一点小的变化,不过如果你了解了编译器做了
什么以后的话,你也就不会太奇怪了。不过关于函数部分的内容我下次再说了。
现在回到上面的内容。我们这里还是讲一下内建类型。显然一样类型的变量是可以互相赋值。
不过当然还有一些其他情况也是可以的,比如类型的宽化,关于类的继承体系问题等等。
当然了,不一样的类型一般来说是不能互相赋值,当然这里的例外就是强制转化,
类的继承体系等情况了。
看到这里就会明白下面的程序为什么会运行的了。
我这里也把下面的程序作为今天内容的总结:
- #include <iostream>
- using namespace std;
- int main()
- {
- int a[2]={10,20};
- int *p=a;//根据上面说明,由于编译器的参与,两者类型转化后一致
- int vInt=10;
- int *parr[2]={&vInt,&vInt};
- int **p2p=parr;//上面分析,类型一致
- return 0;
- }
C++语言中数组指针和指针数组彻底分析的更多相关文章
- 以杨辉三角为例,从内存角度简单分析C语言中的动态二维数组
学C语言,一定绕不过指针这一大难关,而指针最让人头疼的就是各种指向关系,一阶的指针还比较容易掌握,但一旦阶数一高,就很容易理不清楚其中的指向关系,现在我将通过杨辉三角为例,我会用四种方法从内存的角度简 ...
- C语言中如何将二维数组作为函数的参数传递
今天写程序的时候要用到二维数组作参数传给一个函数,我发现将二维数组作参数进行传递还不是想象得那么简单里,但是最后我也解决了遇到的问题,所以这篇文章主要介绍如何处理二维数组当作参数传递的情况,希望大家不 ...
- C语言中的声明解析规则——数组,指针与函数
摘要:C语言的申明存在的最大问题是:你无法以一种人们所习惯的自然方式和从左向右阅读一个声明,在引入voliatile和const关键字以后,情况更加糟糕了.由于这些关键字只能出现在声明中,是的声明形式 ...
- C语言中 指向函数的指针 简介
引子:在学习CPrimerPlus的第十四章的14.13节中,遇到了如下三行文字,是有关指向函数的指针的,把我搞晕了. char * fump(); //返回指向char的指针的函数 char (* ...
- C语言中的函数与指针
1. 为什么需要函数? 函数就是功能的封装. 函数就是为了实现某个功能而编写的一段代码 scanf() , printf() 2.函数优点: 代码更简洁 代码复用 如果业务逻辑变化,只把相应的 ...
- (转)C语言中长度为0的数组
前面在看Xen的源码时,遇到了一段代码,如下所示: 注意上面最后一行的代码,这里定义了一个长度为的数组,这种用法可以吗?为什么可以使用长度为0 的数组?长度为的数组到底怎么使用?……这篇文章主要针对该 ...
- c语言中的结构体指针类型的cast
1.我们在c语言中会经常碰到强制类型转换. 在这,我介绍一种结构pointer类型转换,但是有前提(有点类似于c++中的继承中的子父对象的cast). 简单的介绍一下: 首先我们要知道一个结构的指针, ...
- c语言中指向整型指针的指针的理解
/************************************************************************* > File Name: ptr_ptr_i ...
- c语言中函数调用的本质从汇编角度分析
今天下午写篇博客吧,分析分析c语言中函数调用的本质,首先我们知道c语言中函数的本质就是一段代码,但是给这段代码起了一个名字,这个名字就是他的的这段代码的开始地址 这也是函数名的本质,其实也就是汇编中的 ...
- c语言中如何通过二级指针来操作二维数组
通过二级指针去访问二维数组需要先给二级指针分配等同于二维数组行数的一维数组指针,然后把二维数组的每行首地址赋值给对应位置的一维指针上.之后就可以通过二维指针直接访问了. 参考代码如下,可以看具体注释辅 ...
随机推荐
- [C++] String Basic
Namespace Declarations A using declaration let us use a name from a namespace without qualify the na ...
- POJ 2104 K-th Number(划分树)
Description You are working for Macrohard company in data structures department. After failing your ...
- vue移动音乐app开发学习(二):页面骨架的开发
本系列文章是为了记录学习中的知识点,便于后期自己观看.如果有需要的同学请登录慕课网,找到Vue 2.0 高级实战-开发移动端音乐WebApp进行观看,传送门. 完成后的页面状态以及项目结构如下: 一: ...
- 第一章 Java Web应用开发技术
Java Web应用开发是基于JavaEE(JavaEnterprise Edition)框架的,而JavaEE是建立在Java平台上的企业级应用解决方案.JavaEES框架提供的Web开发技术主要支 ...
- lintcode-148-颜色分类
148-颜色分类 给定一个包含红,白,蓝且长度为 n 的数组,将数组元素进行分类使相同颜色的元素相邻,并按照红.白.蓝的顺序进行排序. 我们可以使用整数 0,1 和 2 分别代表红,白,蓝. 注意事项 ...
- 3dContactPointAnnotationTool开发日志(二)
今天看的时候发现其实www的方式是可以根据指定路径读取本地图片到Image中的.也就是昨天提到的第二种方式. 随便选了个图片做示范: 修改后的代码如下: using System.Collec ...
- my.conf 修改编码
mysql汉字乱码的原因 mysql默认的编码是Latin1是I-8859-1的别名,但Latin1是不支持汉字的,所以要将其改为UTF-8或GBK 1.关闭mysql服务器,这个很重要. 2.通过m ...
- python爬虫从入门到放弃(四)之 Requests库的基本使用(转)
什么是Requests Requests是用python语言基于urllib编写的,采用的是Apache2 Licensed开源协议的HTTP库如果你看过上篇文章关于urllib库的使用,你会发现,其 ...
- (转)Linux常用性能检测命令
一.uptime Uptime命令的显示结果包括服务器已经运行了多长时间,有多少登陆用户和对服务器性能的总体评估(load average).load average值分别记录了上个1分钟,5 ...
- c++读取文件夹及子文件夹数据
这里有两种情况:读取文件夹下所有嵌套的子文件夹里的所有文件 和 读取文件夹下的指定子文件夹(或所有子文件夹里指定的文件名) <ps,里面和file文件有关的结构体类型和方法在 <io.h ...