指针c语言与数组,C语言指针与数组

C语言指针与数组

数组的下标应该从0还是1开始? 我提议的妥协方案是0.5,可惜他们未予认真考虑便一口回绝    -- Stan Kelly-Bootle

1. 数组并非指针

为什么很多人会认为指针和数组始终应该可以互换的呢? 因为对数组的引用总是可以写成对指针的引用,而且确实存在一种指针和数组的定义完全相同的上下文环境,

不幸的是,这只是数组的一种极为普通的用法,并非所用情况下都是如此。

2. 什么是声明,什么是定义

C语言中对象必须有且只有一个定义,但它可以有多个extern声明.

定义:只能出现在一个地方,确定对象的类型并分配内存,用于创建新的对象,例如 int a[100]

声明:可以多次出现,描述对象的类型,用于指代其他地方定义的对象(例如在其他文件里) 例如 extern int a[100]

extern对象声明告诉编译器对象的类型和名字,对象的内存分配则在别处进行

3. 数组与指针的区别

出现在赋值左边的符号被称为 左值, 出现在赋值右边的符号被称为 右值。

编译器为每个变量分配一个地址(左值),这个地址在编译时可知,并且该变量在运行时一直保存于这个地址中。

存储于变量中的值(右值)只有在运行时才可知,如果需要用到变量中存储的值,编译器就发出指令从指定地址读入变量并将它存于寄存器中。

例如: char a[9] = "abcdefgh";   c = a[i]

假设编译器符号表具有一个地址9980

运行时步骤1:取 i 的值,将它与 9980 相加 (基址加偏移量)

运行时步骤2:取地址 [9980+i] 的内容

例如:char* p;  c = *p;

假设编译器符号表有一个符号p,它的地址为4624

运行时步骤1:取地址 4624 的内容,假设是 1024

运行时步骤2:取地址1024的内容

例如:char* p = "abcdefgh"   c = p[i]

假设编译器符号表有一个p,地址为 4624

运行时步骤1:取地址4624的内容,假设是 1024

运行时步骤2:取得 i 的值,并将它与 1024 相加  (基址加偏移量)

运行时步骤3:取地址 [1024+i] 的内容

指针

数组

保存数组的地址

保存数据

间接访问数据,首先取指针的内容,把它作为地址,然后从这个地址提取数据。

如果指针有一个下标[i],就把指针的内容 加上i作为地址,从中提取数据

直接访问数据 a[i]只是简单的以a+i作为地址取数据

通常用于动态数据结构

通常用于存储数目固定且数据类型相同的元素

相关的函数为 malloc free

隐式分配和删除

通常指向匿名数据

自身即为数据名

定义指针时,编译器没有为指针所指向的对象分配空间,只是分配指针本身的空间

ANSI C中,初始化指针时所创建的字符串常量所定义为 只读。在有些编译器中,字符串常量被存放在只允许读取的文本段中,以防止它被修改

char* ptr = "hello world"    // 这种写法是非常不推荐的,因为 ptr 所指对象是只读的,这将隐式的将 const 转 non-const,任何对 ptr 的修改都会 coredump

const char* ptr = "hello world"   // 这种写法明确表示 ptr 的const属性,这时对 ptr 所指对象的修改都会 在编译时报错

char a[] = "hello world"              //这种写法表明 数组中元素都是可修改的,但是数组名是不可修改的左值,即数组首地址不可以改变,是常量

4. 什么时候数组与指针相同

所有作为函数参数的数组名可以通过编译器转换为指针

数组的声明就是数组,指针的声明就是指针,两者不能混淆

注意:如果定义一个数组,在其他文件中对它extern声明时也必须把它声明为数组,指针也是如此

在使用数组(在语句或表达式中引用)时,数组总是可以写成指针的形式

cfa961835fb9c8bf43754bc26cb77041.png

数组下标表达式总是可以改写成带偏移量的指针表达式

什么时候数组和指针是相同的:

规则1:表达式中的数组名(与声明不同)被编译器当作一个指向该数组第一个元素的指针 (具体见 ANSI C标准 第6.2.2.1 节)

规则2:下标总是与指针的偏移量相同(具体见ANSI C标准 第6.3.2.1 节)

规则3:在函数参数的声明中,数组名被编译器当作指向该数组第一个元素的指针(具体见ANSI C标准 第6.7.1 节)

其实规则1和规则1合起来理解如下:对数组下标的引用总是写成"一个指向数组的起始地址的指针加上偏移量"

对数组的引用如a[i]在编译时总是被编译器改写成*(a+i)指针访问的形式

"作为函数参数的数组名"等同于指针

void func(int* arg);

void func(int arg[10]);

void func(int arg[]);

上述三种形式是完全等同的

5. 为什么C语言把数组形参当作指针

把作为形参的数组和指针等同起来是出于效率原因的考虑

在C语言中,所有非数组形式的数据实参均以传值形式(对实参做一份拷贝并传递给调用的函数,函数不能修改作为实参的实际变量的值,而只能修改传递给它的那份拷贝)调用

C语言允许程序员把形参声明为数组(程序员打算传递给函数的东西)或者指针(函数实际所接收到的东西)

不管程序员实际所写的是哪种形式,函数并不自动知道指针所指数组共有多少个元素,所以必须有个约定,如数组以NULL结尾或者另一个附加的参数表示数组的范围

可以通过向函数传递一个指向数组第一个元素的指针来访问整个数组,但也可以让指针指向任何一个元素,这样传递给函数的就是从该元素之后的数组片段

C语言指针和数组知识总结(上)

C语言指针和数组知识总结(上) 一.指针的基础 1.C语言中,变量的值能够通过指针来改变,打印指针的语句符号可以是:  %08x 2.指针的本质 指针的本质就是变量,那么既然是变量,那么一定会分配地址 ...

C语言指针与数组的定义与声明易错分析

部分摘自 1.定义为数组,声明为指针 在文件1中定义: char a[100]; 在文件2中声明: extern char *a; //这样是错误的 这里的extern告 ...

11-C语言指针&一维数组&字符串

一.用指针遍历数组元素 1.最普通的遍历方式是用数组下标来遍历元素 1 // 定义一个int类型的数组 2 int a[4] = {1, 2, 3, 4}; 3 4 int i; 5 for (i = ...

c语言指针,数组

指针:说简单点就是一个地址.例如int *p,p是个变量,里面放的是地址0x0000,同理,每一个指针,不管什么类型,都是地址,也就是空间都是4个字节(32位机). 以此类推,指针也有指向它的指针in ...

C语言指针和数组

#include int main() { /********************************************* * * 指针和数组: * 定义 ...

c语言 指针与数组

关键概念: 1.多个不同类型的指针可以对应同一个地址: 2.(&p)则是这样一种运算,返回一个指针,该指针的值是当时声明p 时开辟的地址,指针的类型是p的类型对应的指针类型: 3.(*p)操作 ...

C语言指针和数组知识总结(下)

一.数组指针: 数组指针就是一个指针,只不过它指向的是一个数组.可以通过如下方式来定义 typedef int Array[5]; //数组类型 Array* m;      //数组定义 还有一种更 ...

C语言核心之数组和指针详解

指针 相信大家对下面的代码不陌生: int i=2; int *p; p=&i;这是最简单的指针应用,也是最基本的用法.再来熟悉一下什么是指针:首先指针是一个变量,它保存的并不是平常的数据,而 ...

指针与数组的区别 —— 《C语言深度剖析》读书心得

原书很多已经写的很清楚很精炼了,我也无谓做无意义的搬运,仅把一些基础和一些我自己以前容易搞混的地方写一下. 1. 意义: 指针: 指针也是一种类型,长度为4字节,其存放的内容只能是一个地址(4字节). ...

随机推荐

Windows和linux的文件共享(1)

今天是我学习linux以来第一次写博客.带着一种激动心情我开始尝试着去完成第一篇博客.从易到难.所以第一次我就主要学习了Windows之间的文件共享. Window之间的文件共享是通过局域网实现的.局 ...

android:persistent属性

application PhoneApp既没有被Broadcast唤醒,也没有被其他service调用,那么是android是通过什么方式来启动PhoneApp,所以就发现了属性android:per ...

js中的写出想jquery中的函数一样调用

1.IIFE: Immediately-Invoked function Expression 函数模块自调用 2.代码实现

unity项目build成webgl时选择生成目录(解决方法)

在unity里点击File>>Build Settings...>>勾选你要生成的Scenes>>选择webgl>>后面Development Buil ...

docker学习系列(二):使用Dockerfile创建自己的镜像

dockerfile可以允许我们自己创建镜像,通过编写里面的下载软件命令,执行docker build 即可生成镜像文件. 初尝dockerfile 新建一个目录test,然后进入这个目录,创建一个名 ...

主引导记录MBR的结构和作用

MBR磁盘分区都有一个引导扇区,称为主引导记录,英文简称为MBR.1. MBR的结构MBR扇区位于整个硬盘的第一个扇区:按照C/H/S地址描述,即0柱面〇磁头1扇 区:按照LBA地址描述即0扇区.它是 ...

Scrum Meeting 5 -2014.11.11

放假过掉一大半.大家都努力赶着进度,算法实现基本完成.可能还有些细小的改动,但也可以统一进入测试阶段了. 今天叫了部分在校人员开了个小会.任务决定以测试为主,同时开始进行服务器的部署. 在之前尝试服务 ...

vue组件中,iview的modal组件爬坑--modal的显示与否应该是使用v-show

这是我第一次写博客,主要是记录下自己解决问题的过程和知识的总结,如有不对的地方欢迎指出来! 需求:点击btn,弹出modal显示图表(以折现图为例) 这应该是很基本的需求也是很容易实现的,代码和效果如 ...

关于Android不同系统版本的市场占比情况详解

一,google官方统计的不同Android版本市场的占比强开 google统计的数据情况 这个是google官方对于不同版本的市场占比情况.这个是针对全世界所有的Android手机占比情况. 二,友 ...

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值