范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文

C语言指针经典知识汇总

  指针在C语言中是一块很重要的内容,也是比较难理解的一块内容,我们需要反复理解反复巩固才可以对其有所了解。之前也分享过指针相关的笔记,但是都比较杂,本篇笔记汇总一下指针相关的内容,包含了挺多指针相关的基础知识点。这篇笔记有点长,可以收藏下来慢慢阅读。复杂类型说明
  以下这部分内容主要来自《让你不再害怕指针》:
  要了解指针,多多少少会出现一些比较复杂的类型,所以,先介绍一下如何完全理解一个复杂类型,要理解复杂类型其实很简单。
  一个类型里会出现很多运算符,他们也像普通的表达式一样,有优先级,其优先级和运算优先级一样,所以我总结了一下其原则: 从变量名处起,根据运算符优先级结合,一步一步分析。
  下面让我们先从简单的类型开始慢慢分析吧:int p;  
  这是一个普通的整型变量 。
  int *p;  
  首先从 P处开始,先与*结合,所以说明 P 是一个指针,然后再与 int 结合,说明指针所指向的内容的类型为 int 型。所以 P 是一个返回整型数据的指针。
  int p[3];  
  首先从 P 处开始,先与[]结合,说明 P 是一个数组,然后与 int 结合,说明数组里的元素是整型的,所以 P 是一个由整型数据组成的数组。
  int *p[3];
  首先从 P 处开始,先与[]结合,因为其优先级比 * 高,所以 P 是一个数组,然后再与 * 结合,说明数组里的元素是指针类型,然后再与 int 结合,说明指针所指向的内容的类型是整型的,所以P 是一个由返回整型数据的指针所组成的数组 。
  int (*p)[3];
  首先从 P 处开始,先与 * 结合,说明 P 是一个指针然后再与[]结合与"()"这步可以忽略,只是为了改变优先级),说明指针所指向的内容是一个数组,然后再与 int 结合,说明数组里的元素是整型的。所以 P 是一个指向由整型数据组成的数组的指针。
  int **p;
  首先从 P 开始,先与后再与 * 结合,说明指针所指向的元素是指针,然后再与 int 结合,说明该指针所指向的元素是整型数据。由于二级以上的指针极少用在复杂的类型中,所以后面更复杂的类型我们就不考虑多级指针了,最多只考虑一级指针。
  int p(int);  
  从 P 处起,先与()结合,说明 P 是一个函数,然后进入()里分析,说明该函数有一个整型变量的参数然后再与外面的 int 结合,说明函数的返回值是一个整型数据。
  int (*p)(int);
  从 P 处开始,先与指针结合,说明 P 是一个指针,然后与()结合,说明指针指向的是一个函数,然后再与()里的int 结合,说明函数有一个 int 型的参数,再与最外层的int 结合,说明函数的返回类型是整型,所以 P 是一个指向有一个整型参数且返回类型为整型的函数的指针。
  说到这里也就差不多了,我们的任务也就这么多,理解了这几个类型,其它的类型对我们来说也是小菜了。不过我们一般不会用太复杂的类型,那样会大大减小程序的可读性,请慎用,这上面的几种类型已经足够我们用了。
  分析指针的方法
  指针是一个特殊的变量, 它里面存储的数值被解释成为内存里的一个地址。要搞清一个指针需要搞清指针的四方面的内容: 指针的类型、 指针所指向的类型、 指针的值(指针所指向的内存区)、 指针本身所占据的内存区。 让我们分别说明。
  先声明几个指针放着做例子:(1)int *ptr; (2)char*ptr; (3)int **ptr; (4)int (*ptr)[3]; (5)int *(*ptr)[4];
  1、指针的类型
  从语法的角度看, 你只要把指针声明语句里的指针名字去掉, 剩下的部分就是这个指针的类型。 这是指针本身所具有的类型。 让我们看看例一中各个指针的类型:(1)int*ptr;//指针的类型是 int* (2)char*ptr;//指针的类型是 char* (3)int**ptr;//指针的类型是 int** (4)int(*ptr)[3];//指针的类型是 int(*)[3] (5)int*(*ptr)[4];//指针的类型是 int*(*)[4]
  2、指针所指向的类型
  当你通过指针来访问指针所指向的内存区时, 指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。
  从语法上看, 你只须把指针声明语句中的指针名字和名字左边的指针声明符*去掉, 剩下的就是指针所指向的类型。例如:(1)int*ptr; //指针所指向的类型是 int (2)char*ptr; //指针所指向的的类型是 char (3)int**ptr; //指针所指向的的类型是 int* (4)int(*ptr)[3]; //指针所指向的的类型是 int()[3] (5)int*(*ptr)[4]; //指针所指向的的类型是 int*()[4]
  在指针的算术运算中, 指针所指向的类型有很大的作用。
  3、指针的值
  指针的值是指针本身存储的数值, 这个值将被编译器当作一个地址, 而不是一个一般的数值。 在 32 位程序里, 所有类型的指针的值都是一个 32 位 整数, 因为 32 位程序里内存地址全都是 32 位长。
  指针所指向的内存区就是从指针的值所代表的那个内存地址开始, 长度为 sizeof(指针所指向的类型)的一片内存区。
  以后, 我们说一个指针的值是 XX, 就相当于说该指针指向了以 XX 为首地址的一片内存区域; 我们说一个指针指向了某块内存区域,就相当于说该指针的值是这块内存区域的首地址。
  指针所指向的内存区和指针所指向的类型是两个完全不同的概念。 在例一中, 指针所指向的类型已经有了, 但由于指针还未初始化, 所以它所指向的内存区是不存在的, 或者说是无意义的。
  以后, 每遇到一个指针, 都应该问问: 这个指针的类型是什么? 指针指向的类型是什么? 该指针指向了哪里? (重点注意) 。
  4、指针本身所占据的内存区
  指针本身占了多大的内存? 你只要用函数 sizeof(指针的类型)测一下就知道了。 在 32 位平台里, 指针本身占据了 4 个字节的长度。指针本身占据的内存这个概念在判断一个指针表达式(后面会解释) 是否是左值时很有用。
  指针的算术运算
  指针可以加上或减去一个整数。 指针的这种运算的意义和通常的数值的加减运算的意义是不一样的, 以单元为单位。
  这在内存上体现为:相对这个指针向后偏移多少个单位或向前偏移了多少个单位,这里的单位与指针变量的类型有关。在32bit环境下,int类型占4个字节,float占4字节,double类型占8字节,char占1字节。
  【注意】一些处理整数的操作不能用来处理指针。例如,可以把两个整数相乘,但是不能把两个指针相乘。示例程序#include   int main(void) { 	int    a = 10, *pa = &a; 	float  b = 6.6, *pb = &b; 	char   c = "a", *pc = &c; 	double d = 2.14e9, *pd = &d;  	//最初的值 	printf("pa0=%d, pb0=%d, pc0=%d, pd0=%d ", pa, pb, pc, pd); 	//加法运算 	pa += 2;  	pb += 2;  	pc += 2; 	pd += 2; 	printf("pa1=%d, pb1=%d, pc1=%d, pd1=%d ", pa, pb, pc, pd); 	//减法运算 	pa -= 1;  	pb -= 1;  	pc -= 1; 	pd -= 1; 	printf("pa2=%d, pb2=%d, pc2=%d, pd2=%d ", pa, pb, pc, pd);  	return 0; }
  运行结果为:pa0=6422268, pb0=6422264, pc0=6422263, pd0=6422248 pa1=6422276, pb1=6422272, pc1=6422265, pd1=6422264 pa2=6422272, pb2=6422268, pc2=6422264, pd2=6422256
  解析:
  举例说明pa0→pa1→pa2的过程,其他类似。pa0+2*sizeof(int)=pa1,pa1-1*sizeof(int)=pa2。因为pa为int类型的指针,所以加减运算是以4字节(即sizeof(int))为单位地址向前向后偏移的。看下图:
  如图:pa1所指向的地址在pa0所指向地址往后8字节处,pa2指向地址在pa1指向地址往前4字节处。
  从本示例程序中,还可以看出:连续定义的变量在内存的存储有可能是紧挨着的,有可能是分散着的。
  数组和指针的联系
  数组与指针有很密切的联系,常见的结合情况有以下三种:数组指针指针数组二维数组指针
  1、数组指针
  数组指针:指向数组的指针。如:int arr[] = {0,1,2,3,4}; int *p = arr; //也可写作int *p=&arr[0]
  也就是说,p,arr,&arr[0]都是指向数组的开头,即第0个元素的地址。
  如果一个指针p指向一个数组arr[]的开头,那么p+i为数组第i个元素的地址,即&arr[i],那么*(p+i)为数组第i个元素的值,即arr[i]。
  同理,若指针p指向数组的第n个元素,那么p+i为第n+1个元素的地址;不管 p 指向了数组的第几个元素,p+1 总是指向下一个元素,p-1 也总是指向上一个元素。
  下面示例证实了这一点:#include   int main(void) {    int arr[] = {0, 1, 2, 3, 4};    int *p = &arr[3];  //也可以写作 int *p = arr + 3;     printf("%d, %d, %d, %d, %d ",     *(p-3), *(p-2), *(p-1), *(p), *(p+1) );    return 0; }
  运行结果为:0, 1, 2, 3, 4
  2、指针数组
  指针数组:数组中每个元素都是指针。如:int a=1,b=2,c=3; int *arr[3] = {&a,&b,&c};
  示例程序:#include  int main(void) {    int a = 1, b = 2, c = 3;    //定义一个指针数组    int *arr[3] = {&a, &b, &c};//也可以不指定长度,直接写作 int *parr[]    //定义一个指向指针数组的指针    int **parr = arr;    printf("%d, %d, %d ", *arr[0], *arr[1], *arr[2]);    printf("%d, %d, %d ", **(parr+0), **(parr+1), **(parr+2));     return 0; }
  第一个 printf() 语句中,arr[i] 表示获取第 i 个元素的值,该元素是一个指针,还需要在前面增加一个 * 才能取得它指向的数据,也即 *arr[i] 的形式。
  第二个 printf() 语句中,parr+i 表示第 i 个元素的地址,*(parr+i) 表示获取第 i 个元素的值(该元素是一个指针),**(parr+i) 表示获取第 i 个元素指向的数据。
  指针数组还可以和字符串数组结合使用,请看下面的例子:#include  int main(void) {     char *str[3] =      {         "hello C",         "hello C++",         "hello Java"     };     printf("%s %s %s ", str[0], str[1], str[2]);     return 0; }
  运行结果为:hello C hello C++ hello Java
  3、二维数组指针
  二维数组指针:指向二维数组的指针。如:int a[3][4] = { {0, 1, 2, 3}, {4, 5, 6, 7}, {8, 9, 10, 11} }; int (*p)[4] = a;
  a [3] [4]表示一个3行4列的二维数组,其所有元素在内存中是连续存储的。
  请看如下程序:#include  int main(void) {     int a[3][4] = { {0, 1, 2, 3}, {4, 5, 6, 7}, {8, 9, 10, 11} };     int i,j;     for( i = 0; i < 3; i++ )     {         for( j = 0; j < 4; j++ )         {             printf("a[%d][%d]=%d ", i, j, &a[i][j]);         }     }      return 0; }
  运行结果为:a[0][0]=6422216 a[0][1]=6422220 a[0][2]=6422224 a[0][3]=6422228 a[1][0]=6422232 a[1][1]=6422236 a[1][2]=6422240 a[1][3]=6422244 a[2][0]=6422248 a[2][1]=6422252 a[2][2]=6422256 a[2][3]=6422260
  可见,每个元素的地址都是相差4个字节,即每个连续在内存中是连续存储的。
  按照以上定义可归纳出如下4个结论:
  (1)p指向数组a的开头,也即第1行;p+1前进一行,指向第2行。
  (2)*(p+1)表示取第2行元素(一整行元素)。
  (3)*(p+1)+1表示第2行第2个元素的地址。
  (4)((p+1)+1)表示第2行第2个元素的值。
  综上4点,可得出如下结论:a+i == p+i  *(a+i) == *(p+i) a[i][j] == p[i][j] == *(a[i]+j) == *(p[i]+j) == *(*(a+i)+j)== *(*(p+i)+j)
  以上就是数组与指针常用的三种结合形式。
  指针与数组的区别
  数组与指针在多数情况是可以等价的,比如:int array[10]={0,1,2,3,4,5,6,7,8,9},value; value=array[0]; //也可写成: value=*array; value=array[3]; //也可写成: value=*(array+3); value=array[4]; //也可写成: value=*(array+4)
  但也有不等价的时候,比如如下三种情况:数组名不可以改变,而指向数组的指针是可以改变的。字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。求数组长度时,借用数组名可求得数组长度,而借用指针却得不到数组长度。
  1、区别一
  数组名的指向不可以改变,而指向数组的指针是可以改变的。
  请看如下代码:#include   int main(void) {     int a[5] = {0, 1, 2, 3, 4}, *p = a;     char i;      // 数组遍历方式一     for ( i = 0; i < 5; i++ )     {         printf("a[%d] = %d ", i, *p++);     }      // 数组遍历方式二     for ( i = 0; i < 5; i++ )     {         printf("a[%d] = %d ", i, *a++);     }      return 0; }
  数组遍历方式一:使用指针遍历数组元素,* p++等价于*(p++),即指针指向的地址每次后移一个单位,然后再取地址上的值。这里的一个单位是sizeof(int)个字节。
  数组遍历方式二:使用数组名自增遍历数组元素,编译出错,错误如下:error: value required as increment operand
  因为数组名的指向是不可以改变的,使用自增运算符自增就会改变其指向,这是不对的,数组名只能指向数组的开头。但是可以改为如下遍历方式:for ( i = 0; i < 5; i++ ) {     printf("a[%d] = %d ", i, *(a+i)); }
  这可以正确遍历数组元素。因为*(a+i)与a[i]是等价的。
  2、区别二
  字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。
  请看如下代码://字符串定义方式一 char str[] = "happy";  //字符串定义方式二 char *str = "happy";
  字符串定义方式一:字符串中的字符是可以改变的。如可以使用类似str[3]="q"这样的语句来改变其中的字符。原因就是:这种方式定义的字符串保存在全局数据区或栈区,是可读写的。
  字符串定义方式二:字符串中的字符是不可以改变的。原因就是:这种方式定义的字符串保存在常量区,是不可修改的。
  2、区别三
  求数组长度时,借用数组名可求得数组长度,而借用指针却得不到数组长度。
  请看如下代码:#include   int main(void) {     int a[] = {0, 1, 2, 3, 4}, *p = a;     char len = 0;      // 求数组长度方式一     printf("方式一:len=%d ",sizeof(a)/sizeof(int));      // 求数组长度方式二     printf("方式二:len=%d ",sizeof(p)/sizeof(int));      return 0; }
  运行结果方式一:len=5 方式二:len=1
  求数组长度方式一:借用数组名来求数组长度,可求得数组有5个元素,正确。
  求数组长度方式二:借用指针求数组长度,求得长度为1,错误。原因是:
  p只是一个指向int类型的指针,编译器不知道其指向的是一个整数还是指向一个数组。sizeof(p)求得的是p这个指针变量本身所占用的字节数,而不是整个数组占用的字节数。
  下面还需要注意数组名的一个问题: 声明了一个数组 TYPE array[n] , 则数组名是一个常量指针, 该指针的值是不能修改的, 即类似 array++的表达式是错误的。
  指针函数与函数指针
  函数、指针这两个词结合的顺序不同其意义也不同,即指针函数与函数指针的意义不同。
  1、指针函数
  指针函数的本质是一个函数,其返回值是一个指针。示例如下:int *pfun(int, int);
  由于"*"的优先级低于"()"的优先级,因而pfun首先和后面的"()"结合,也就意味着,pfun是一个函数。即:int *(pfun(int, int));
  接着再和前面的"*"结合,说明这个函数的返回值是一个指针。由于前面还有一个int,也就是说,pfun是一个返回值为整型指针的函数。
  指针函数示例程序如下:#include  //这是一个指针函数的声明 int *pfun(int *arr, int n);  int main(void) {     int array[] = {0, 1, 2, 3, 4};     int len = sizeof(array)/sizeof(array[0]);     int *p;     int i;      //指针函数的调用     p = pfun(array, len);      for (i = 0; i < len; i++)     {         printf("array[%d] = %d ", i, *(p+i));     }      return 0; }  //这是一个指针函数,其返回值为指向整形的指针 int *pfun(int *arr, int n) {     int *p = arr;      return p; }
  程序运行结果如下:
  主函数中,把一个数组的首地址与数组长度作为实参传入指针函数pfun里,把指针函数的返回值(即指向数组的指针)赋给整形指针p。最后使用指针p来遍历数组元素并打印输出。
  2、函数指针
  函数指针其本质是一个指针变量,该指针变量指向一个函数。C程序在编译时,每一个函数都有一个入口地址,该入口地址就是函数指针所指向的地址。函数指针示例:/*声明一个函数指针 */ int (*fptr) (int, int);  /* 函数指针指向函数func */ fptr = func;  // 或者fptr = &func;
  func是一个函数名,那么func与&func都表示的是函数的入口地址。同样的,在函数的调用中可以使用:方式一:func(),也可以使用方式二:(*fun)()。这两种调用方式是等价的,只是我们平时大多都习惯用方式一的调用方法。
  至于为什么func与&func的含义相同,《嵌入式Linux上的C语言编程实践》这本书中有如下解释:
  对于函数func来说,函数的名称就是函数代码区的常量,对它取地址(&func)可以得到函数代码区的地址,同时,func本身也可以视为函数代码区的地址。因此,函数名称和对其取地址其含义是相同的。
  函数指针示例程序如下:#include   int add(int a, int b);  int main(void) {     int (*fptr)(int, int); //定义一个函数指针     int res;     fptr = add;  //函数指针fptr指向函数add      /* 通过函数指针调用函数 */     res = (*fptr)(1,2); //等价于res = fptr(1,2);     printf("a + b = %d ", res);      return 0; }  int add(int a, int b) {     return a + b; }
  程序运行结果如下:
  以上就是关于指针函数与函数指针的简单区分。其中,函数指针广泛应用于嵌入式软件开发中,其常用的两个用途:调用函数和做函数的参数。
  以上就是本次的分享,如有错误,欢迎指出!谢谢

网友某鱼捡漏入手iPhone12ProMax,换个电池还能再战三年随着iPhone13系列热卖和降价,iPhone12系列价格也在持续走低。特别是屏幕大性能好的iPhone12ProMax性价比非常高。网友在某鱼捡漏入手了一台iPhone12Pr折叠手机一定会有折痕,那消费者该如何选择?一篇文章告诉你Find镜面无折痕?Fold3技术太落后?OPPOFindN发布后,很多人对这款折叠屏手机的折痕控制赞不绝口,刘作虎在发布会上也宣称,这是目前市面上折痕最小的折叠手机,在屏幕完全展磁盘上怎么存储NULL值?1为何不能直接存个NULL?NULL值列表,一行数据里可能有的字段值是NULL,比如nickname字段,允许为NULL,存储时,如果没赋值,这字段值就是NULL。假设这个字段的NPython学习笔记(二)变量和简单数据类型(一)运行helloworld。py时发生的情况末尾py指出这是一个Python程序,因此编辑器使用Python解释器来运行它变量messageHelloPythonworld!prin热点解读1。春节电影2。数字货币3。新能源车4。巴菲特春节电影春节电影开始失宠,近10年票房首次出现倒退。点评今年春节档票房都显著落后去年同期,港股复市,传媒股阿里影业猫眼娱乐纷纷下跌。随着观影人数的显著回落,即使现在影院开始降价,今硅谷始料未及!法国拆下3000根华为天线后,态度再次发生了变化大家都知道,在国内的通信行业来说,华为是咱们通信行业的龙头。经过了这几十年的发展,华为现在取得了很不错的成绩。众所周知,华为能够取得现在的成就,并不是一蹴而就的,华为终于开始一个,我,阿里的普通员工,工作5年,透露员工的年终奖金有多少?近年来,不少大学生在毕业之后找工作的过程中都会发现,一些企业都在不断的进行相应的招聘,无论是国企还是私企,大家都会给员工较高的福利待遇,希望能够通过较高的奖金来吸引人才,而作为大学PC端电脑硬件市场已死?(中篇)现在上大学,一部智能手机已经是必备了吧?网课时代的大学,电脑也少不了吧?距离今天的十几年前,比如我经历的那个时代2008年。大学宿舍里,能有一台上网本(神舟笔记本不超过3000元)买电脑为什么不能去实体店?网上买省事省力,去外面电脑店买还要跟他们斗智斗勇,烦不烦,这个互联网发达的年代,网购比零售价便宜多了,当然其实很多都是不太懂电脑的,去网上买是一脸懵逼也没人给个推荐但是很多人都是电想买一台台式电脑,2000以下有啥推荐?2000元没有合适的,旧的差不多,但是可以组装一个cpui310105主板微星h510爆破弹显卡自带核显内存威刚万紫千红26668g一根主硬盘金士顿A2000500g散热器冰凌电源能推荐一台能打吃鸡与gta5的笔记本电脑吗,六七千的,谢谢?我从07年开始买游戏本,无一例外地让我失望,高热是最主要的问题,低帧数也是必然。笔记本里的CPU和gpu都是移动版,低功耗版。rtx3060与rtx3060m差别巨大。比同型号台式
世界上最短的电梯,只有四五层台阶的高度,并获吉尼斯世界纪录世界上最短的电梯在哪?呶!位于日本川崎市商场的这个号称全世界最短的电梯,整个电梯五个台阶,全长只有83cm,呆萌的外表,一度也被称为卖萌电梯。可这么多年它也没停止运营,网友们可坐不华山东峰上风光与赌棋亭华山东峰位于陕西省渭南市华阴市,又名朝阳峰,海拔2090米,是华山主峰之一,因位置居东得名。峰顶有一平台,居高临险,视野开阔,是著名的观日出的地方,人称朝阳台,东峰也因之被称为朝阳华山东峰朝阳峰风光华山东峰位于陕西省渭南市华阴市,又名朝阳峰,海拔2090米,是华山主峰之一,因位置居东得名。峰顶有一平台,居高临险,视野开阔,是著名的观日出的地方,人称朝阳台,东峰也因之被称为朝阳科普电动汽车终局之战固态电池最近大众汽车集团召开了隆重的线上电池日PowerDay,其中公布了包括固态电池在内的很多电动汽车电池技术细节。很多朋友关心什么是电动汽车固态电池,为什么固态电池被称为电动汽车的终局WiFi相机FCC认证办理WiFi相机FCC认证办理步骤介绍。Wifi数码相机即相机带wifi功能,可以与wifi连接,让能通过网络更加便捷的上传或分享照片。但一般这类产品要出口美国则必须要办理FCC认证。做什么生意最赚钱?做什么生意最赚钱?有这样一个故事从前有一个小镇,每个人都喜欢借钱,平时就靠信用卡来过日子。有一天来了一位有钱的外地旅客,他来旅店住宿,拿出1000块钱放在了柜台上,说想选一间合适的神来之笔9毛9的传奇故事做面条起家,年入20亿,上市之后股价暴涨百分之44。这简直就是神来之笔。在港股上市的9毛9集团,人称一块不到,目前他们集团旗下的9毛9在中国西北菜排名里面排老二,而泰二这个品牌在酸赚钱那些事儿1(小说连载)赚钱的秘密是迷恋一个人1hr有些人很自恋,其实长丑点没关系,但长的丑还一天到晚的自恋那就一定是到达了至高的境界。大家仔细观察一下,这个世界上只要长的丑的人,同时又位高权重的,这样的怎么赚到第一个1000万?怎么赚到第一个1000万?现在到处自媒体平台在宣传1000万有多好挣,多么容易,导致现在月入5000的人都很自卑,不敢说话。1000万到底有多少?1个月挣10万,减去别的开销存8万海外捞金日入300刀你绝对想知道的秘密海外捞金日入300刀你绝对想知道的秘密如果你是个小白,一点互联网项目经验都没有,那就不要做一些收益高又轻松的项目,因为你经不起那些诱惑,下场就是被割割割,还是做一些脚踏实地的项目。赚钱背后的真相赚钱背后的真相全世界的富人占总数的3,中产阶级占10,87(含87)以下的人是穷人,比例为19,所有的穷人都在想尽办法变成富人。韩国电影寄生虫讲述两种穷人一种甘做富人的寄生虫,另一