1.3 sizeof和strlen的对比 二、数组和指针笔试题解析 图示指针3文章的内容,这里会用到 还需要指针1内存的知识 2.1 一维数组 int main() { int a[] = 注意:数组的地址就是数组最开始位置的地址,数组首元素地址也是那个地址,地址就是指针,在指针1中写过,指针变量的大小与类型无关,就是4/8个字节,只与运行环境有关。 (arr));//6, arr表示整个数组,sizeof(arr)计算的是整个数组的大小 //6个元素,每个元素是一个char,一个char就是一个字节,一共占6个字节 printf(" ,大小就是4/8个字节 跳过之后(&arr+1)和&arr的类型是一样的,跳过之后还是一个指向6个元素的一个数组的地址,因为&arr取出的就是一个6个字符的数组的地址,+1之后类型不变,还是一个数组指针 总结 以上就是指针6的内容了,指针7将继续写一些指针运算相关的笔试题,不得不说期末周考完就是爽啊,安安心心学编程搞文章舒服多了。喜欢的靓仔靓女们不要忘记一键三连给予支持哦~
一、数组指针 初学C语言的朋友对数组指针这指针数组感到迷惑,分不清,包括我自己,其实是对概念的不清晰以及对数组和指针这两个概念理解的不够深入,下面谈谈我的理解。 数组指针,是一个指针而不是数组。 这个指针具有指向整个数组的能力,保存这个数组的其实地址。 是数组指针变量 二、指针数组 指针数组,是一个数组,而不是指针。 ,这个指针指向.rodata对应的常量字符串 指针数组的应用 完整的main函数原型,int main(int arc,char* argv[],char* envp[]) 其中,两个数组分别保存命令行参数和环境变量 为了提高程序执行的效率,C语言不做数组下标的安全性检查。如果进行了检查当数组数据量非常大时候会显著减低程序的效率,在安全性和高效率之间,权衡利弊之下。
int a1,a2,a3; p[0]=&a1; p[1]=&a2; p[2]=&a3; 在理解: 数组指针(也称行指针) 定义 int (*p)[n]; ()优先级高,首先说明p是一个指针,指向一个整型的一维数组 如要将二维数组赋给一指针,应这样赋值: int a[3][4]; int (*p)[4]; //该语句是定义一个数组指针,指向含4个元素的一维数组。 指针数组 定义 int *p[n]; []优先级高,先与p结合成为一个数组,再由int*说明这是一个整型指针数组,它有n个指针类型的数组元素。 这样两者的区别就豁然开朗了,数组指针只是一个指针变量,似乎是C语言里专门用来指向二维数组的,它占有内存中一个指针的存储空间。指针数组是多个指针变量,以数组形式存在内存当中,占有多个指针的存储空间。 还需要说明的一点就是,同时用来指向二维数组时,其引用和用数组名引用都是一样的。
一、指针数组 解释:一个数组,若其元素均为指针类型数据,称为指针数组 一般形式 类型名 *数组名[数组长度] 二、指向指针数据的指针 解释:指向指针数据的指针声明的是指针,只是这指针指向另一个指针 例子 三、指针数组作main函数的形参 int mian(int argc,char *argv[]) 其中argc和argv就是mian函数的参数,他们是程序的“命令行参数” 命令行的一般形式 命令名 参数 1 参数2 ......参数n 注意:命令名和各参数之间用空格分隔开
01 什么是指针数组 1、一个数组,若其元素均为指针类型数据,称为指针数组,也就是说,指针数组中的每一个元素都存放一个地址,相当于一个指针变量。 2、一般形式 类型名 *数组名[数组长度]; 类型中应包括符合*,如int *表示是指向整数数据的指针类型。 3、例子 int *p[4]; 4、由于[]比*优先级高,因此p先和[4]结合,形成p[4]形式,这显然是数组形式,表示p数组有4个元素。 然后再和p前面的*结合,*表示此数组是指针类型的,每个数组元素都可以指向一个整型变量。 02 指向指针数据的指针 1、在了解了指针数组的基础上,需要了解指向指针数据的指针变量,简称指向指针的指针。 2、在某些情况下,main函数可以有参数,例如:int main(int argc,char *argv[])其中,argc和argv就是main函数的形参,他们是程序的“命令行参数”。
,而前面的 int 是指针变量p 要指向的变量的类型; 例如: #include<stdio.h> int main() { int *p,i;//定义整形指针变量p,和普通变量i; p= ,所以*自然就是指针变量的标志了,在定义的同时,指向一个变量,其实就是把定义和取地址,合并了起来,这两个代码是等同的; 那接下来我们开始了解通过指针访问变量地址,话不多说上代码: 如图:我将普通变量的地址放在了指针变量 ; 数组指针 数值指针,其主体是指针,他就是个指针,不过是有点不同而已,这个指针指向的是数组的地址,在此之前我们需要了解数组的地址; 数组的地址; 我们通常说数组的地址是数组名,是数组的首元素地址,也确实是这样 3.打印的第一行和打印的第3行的地址是相同的,说明数组名就是数组首元素的地址,而对数组取地址的这个地址也同样是数组首元素的地址: 结论:二者都是数组首元素的地址,不同的是,步长大小不同; 我们再回到数组指针上 指针数组 含义 指针数组,主体是数组,不同的是里面存的是指针,是地址; 指针数组打印二维数组 值得注意的是指针数组的[]前面的*p是没有括号的,我们可以这样看,他是一个数组,数组名是p,数组元素个数是3
2-2 误区: 2-3 代码一和代码二的异同: 2-4 关于字符常量区: 2-5 一道为了区分栈区和字符常量区&&字符数组和字符指针的面试题: 3.指针数组 3-1 指针数组长什么样捏? 数组指针 4-1 区分取地址数组名和数组名(老生常谈了) 4-2 辨析数组指针和指针数组 4-3 学会了? 4-2 辨析数组指针和指针数组 图图我就省略喽! ); return 0; } 上面目前讲的基本还是数组指针指向的是一个一维数组 其实数组指针还是可以指向一个二维数组: int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6 我总结的一点小规律 如果光要给上面这个东西命名的话,只用关注*和[]的出现次序排列就可以了 比如;数组指针数组:[]*[] 指针数组指针*[]* 然后再把[]存的内容和*
文章目录 总结 一、使用 数组指针类型 定义数组指针 二、完整代码示例 总结 // 首先 , 定义 数组指针类型 别名 typedef int(*ArrayPointer)[3]; // 然后 , 声明一个 数组指针类型 变量 ArrayPointer p = NULL; 一、使用 数组指针类型 定义数组指针 ---- 使用 数组指针类型 定义数组指针 , 首先 , 使用 typedef 定义一个数组指针类型 , typedef int(*ArrayPointer)[3]; 然后 , 定义一个普通数组 , 之后的 数组指针 指向该数组 , int array2 [3] = {0}; 最后 , 声明一个 数组指针类型 变量 , 将 array2 变量地址赋值给该 数组指针类型 变量 , 指针指向的数据类型为 int[3] 数组类型的变量 array2 ; , 数组元素是指针 (作为参考) char *array = {"12", "ab", "34"}; // 数组指针 , 使用指针变量指向数组 // 使用 数组指针类型 定义数组指针
指针数组 :就是指针的数组,数组的元素是指针; 数组指针:就是指向数组的指针。 简单举例说明: int *p1[10]; 声明了一个数组,数组的元素是int型的指针。 第一种写法:p先和[]结合,所以是一个数组,后与*结合,是指针数组。 第二种写法:()的优先级比[]高,*号和p2构成一个指针的定义,指针变量名为p,int 修饰的是数组的内容,即数组的每个元素。 数组在这里并没有名字,是个匿名数组, 那现在我们清楚p 是一个指针,它指向一个包含10 个int 类型数据的数组,即数组指针 ? int a[3]={1,2,3}; int (*p)[3]=&a;//指向3个int型数组元素的数组指针 int* p2[3]; //存贮3个int型变量地址 for(int i=0;i<3 ;i++) { p2[i]=&a[i]; cout<<(*p)[i]<<" "<<*p2[i]<<endl; } int b[2][3]={{1,2,3},{4,5,6}
下面的摘取的例子特别具有说明性: 以后再也不敢说指针和数组一样啦!
01什么是指针数组 1、一个数组,若其元素均为指针类型数据,称为指针数组,也就是说,指针数组中的每一个元素都存放一个地址,相当于一个指针变量。 2、一般形式 类型名 *数组名[数组长度]; 类型中应包括符合*,如int *表示是指向整数数据的指针类型。 3、例子 int *p[4]; 4、由于[]比*优先级高,因此p先和[4]结合,形成p[4]形式,这显然是数组形式,表示p数组有4个元素。 然后再和p前面的*结合,*表示此数组是指针类型的,每个数组元素都可以指向一个整型变量。 02指向指针数据的指针 1、在了解了指针数组的基础上,需要了解指向指针数据的指针变量,简称指向指针的指针。 2、在某些情况下,main函数可以有参数,例如:int main(int argc,char *argv[])其中,argc和argv就是main函数的形参,他们是程序的“命令行参数”。
一、数组笔试题解析 知识点: 只要是地址就是4/8个字节 数组名 数组名是数组首元素的地址 但是有2个例外: 1. sizeof(数组名) - 数组名表示整个数组,计算的是整个数组的大小 &数组名 - 数组名也表示整个数组,取出的是整个数组的地址 除了这个2个例外,你见到的所有的数组名都表示首元素的地址 sizeof sizeof 是计算对象或者类型创建的对象所占内存空间的大小 printf( "%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]); return 0; } 题目解析及运行结果: ---- 笔试题6: int main() { int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; int *ptr1 = (int *)(&aa + 1); ,如果小伙伴还有不理解的内容,可以参考之前关于指针进阶的博文。
函数指针和其类似,只不过其指向的不是一个变量,而是一个函数,仅此而已。话不多说,看个例子。 简单的5行代码就完成了一个简单的函数指针。我们一行行来看。 第一行。声明了一个函数指针。其和普通指针有啥区别。我放在一起让大家看一下。 void (*funcPointer)(double x);//函数指针 void *pointer;//普通指针 看出来了么。函数指针无非多了两个括号和一个参数列表。想想也是。 所以说函数指针是个很好用的东西。 2. 函数指针数组 函数指针说完了。我们接下来来看看函数指针数组。 多了俩字。数组。前面四个字还一样。那是不是就是有一个数组,用它来盛放函数指针。 好了,相信讲到这里,大家应该清楚的明白什么是函数指针和函数指针数组了。
多维数组 和 多维指针 1. ; 2.数组示例 : int array[6]; 定义数组 int array[6]; 意义 : 数组中包含 6 个 int 类型的数据 , 数组中每个元素都是 int 类型的 ; 第一个元素地址 /6. 定义数组指针, 但是这个赋值过程中 左右两边类型不一致, // array_3 会被强转为 char[6] 类型的数组指针 char(*p2)[6] = array_3; //( 1 ) sizeof 计算大小 和 & 获取地址时 ; 5.具体的数据值存放 : 二维数组第一维是 数组指针, 第二围才是具体的数据值 ; 6.二维数组图示 : 一些注意点 : 1.编译器没有二维数组概念
文章目录 总结 一、直接定义 数组指针 二、完整代码示例 总结 int (*p)[3] = NULL; 一、直接定义 数组指针 ---- 直接定义 数组指针 , 首先 , 定义一个普通数组 , 数组指针类型 变量 , 指针指向的数据类型为 int[3] 数组类型的变量 array2 ; ArrayPointer p = NULL; p = &array2; 验证上述 定义的数组指针 ; } 使用 数组指针 , 打印数组元素内容 : // 使用 数组指针 访问数组中的值 for(i = 0; i < 3; i++) { printf(" , 数组元素是指针 (作为参考) char *array = {"12", "ab", "34"}; // 数组指针 , 使用指针变量指向数组 // 首先 , 定义一个普通数组 , 将 array2 变量地址赋值给该 数组指针类型 变量 // 指针指向的数据类型为 int[3] 数组类型的变量 array2 p = &array2; // 为数组赋值
普通变量和指针变量 共性 PS: 可见这4个函数的汇编指令完全一致,无论是什么类型的指针变量,对指针变量的读写跟普通变量没有任何区别,所谓的指向只是描述指针变量的值时多少而已,就读写而言,指针变量跟普通变量没有任何区别 空指针和野指针 野指针:定义了一个指针变量,如果没有进行初始化,系统就会有可能随机赋值一个地址给这个指针变量,也就是说,这个指向指向一个未知的区域。 图片 图片 PS: 区分指针数组int *a[3]和数组指针int (*a)[3],前者时存放指针的数组,后者是指向数组的指针。 指针数组是多个指针变量,以数组形式存在内存当中,占有多个指针的存储空间。 还需要说明的一点就是,同时用来指向二维数组时,其引用和用数组名引用都是一样的。 ()”的优先级比“[]”高,“*”号和p2 构成一个指针的定义,指针变量名为p2,int 修饰的是数组的内容,即数组的每个元素。
普通变量和指针变量 共性 PS: 可见这4个函数的汇编指令完全一致,无论是什么类型的指针变量,对指针变量的读写跟普通变量没有任何区别,所谓的指向只是描述指针变量的值时多少而已,就读写而言,指针变量跟普通变量没有任何区别 空指针和野指针 野指针:定义了一个指针变量,如果没有进行初始化,系统就会有可能随机赋值一个地址给这个指针变量,也就是说,这个指向指向一个未知的区域。 PS: 区分指针数组int *a[3]和数组指针int (*a)[3],前者时存放指针的数组,后者是指向数组的指针。 还需要说明的一点就是,同时用来指向二维数组时,其引用和用数组名引用都是一样的。 号和p2 构成一个指针的定义,指针变量名为p2,int 修饰的是数组的内容,即数组的每个元素。
int *arr[5];//存放整型地址的指针数组 char *arr[6];//存放字符类型的指针数组 数组指针 概念辨析 我们类比一下: 整型指针:指向整型变量的指针,存放整型变量的地址的指针。 形式辨析 int * p; 首先*表示这是一个指针,命名为p,然后指向的是int类型的指针,数组指针也一样 int(*p) [5]; 上面的形式就是数组指针,我们需要先用()把*和指针名括起来,然后剩下的就是指针指向的类型 数组指针一般和二维数组联动。 因为数组指针存放的是一整个数组的地址,而二维数组传参,二维数组的数组名就表示这个二维数组首行的地址(注意是首行的地址!) [3][5] = { 1,2,3,4,5,2,3,4,5,6,3,4,5,6,7 }; print(arr, 3, 5);//二维数组传参表示二维数组第一行的地址 return 0; } 二维数组的传参 形参接收分为数组接收和指针接收。
. // #include <stdio.h> int main() { int arr[4][4] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15}; =%d\t",i,j,*(p2[i]+j)); } printf("\n"); } return 0; } 运行结果: 从结果可以看出我们成功的使用数组指针和指针数组的方式访问了二位数组 在分析数组指针和指针数组是如何访问二维数组之前,我们通过下面一段代码来学习一下表示二维数组每行起始地址的方式。 . // #include "stdio.h" int main(){ int arr[4][4] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15}; 下面接着分析数组指针和指针数组是如何访问二维数组的,先看数组指针的访问方式。
指针数组与数组指针详解 1.什么是指针数组和数组指针? 2.2数组指针 首先来定义一个数组指针,既然是指针,名字就叫pa char (*pa)[4]; 1 如果指针数组和数组指针这俩个变量名称一样就会是这样:char *pa[4]和char (* 既然pa是一个指针,存放一个数组的地址,那么在我们定义一个数组时,数组名称就是这个数组的首地址,那么这二者有什么区别和联系呢? 因为a是数组首元素首地址,pa存放的却是数组首地址,a是char 类型,a+1,a的值会实实在在的加1,而pa是char[4]类型的,pa+1,pa则会加4,虽然数组的首地址和首元素首地址的值相同,但是两者操作不同 如果是向子函数传参,这和传递一个普通数组的思想一样,不能传递整个数组过去,如果数组很大,这样内存利用率很低,所以应该传递数组的首地址,用一个指针接收这个地址。