指针在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 <stdio.h>
​
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=%dn", pa, pb, pc, pd);//加法运算pa += 2; pb += 2; pc += 2;pd += 2;printf("pa1=%d, pb1=%d, pc1=%d, pd1=%dn", pa, pb, pc, pd);//减法运算pa -= 1; pb -= 1; pc -= 1;pd -= 1;printf("pa2=%d, pb2=%d, pc2=%d, pd2=%dn", 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 <stdio.h>
​
int main(void)
{int arr[] = {0, 1, 2, 3, 4};int *p = &arr[3];  //也可以写作 int *p = arr + 3;
​printf("%d, %d, %d, %d, %dn", *(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 <stdio.h>
int main(void)
{int a = 1, b = 2, c = 3;//定义一个指针数组int *arr[3] = {&a, &b, &c};//也可以不指定长度,直接写作 int *parr[]//定义一个指向指针数组的指针int **parr = arr;printf("%d, %d, %dn", *arr[0], *arr[1], *arr[2]);printf("%d, %d, %dn", **(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 <stdio.h>
int main(void)
{char *str[3] = {"hello C","hello C++","hello Java"};printf("%sn%sn%sn", 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 <stdio.h>
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]=%dn", 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 <stdio.h>
​
int main(void)
{int a[5] = {0, 1, 2, 3, 4}, *p = a;char i;
​// 数组遍历方式一for ( i = 0; i < 5; i++ ){printf("a[%d] = %dn", i, *p++);}
​// 数组遍历方式二for ( i = 0; i < 5; i++ ){printf("a[%d] = %dn", i, *a++);}
​return 0;
}

数组遍历方式一:使用指针遍历数组元素,* p++等价于*(p++),即指针指向的地址每次后移一个单位,然后再取地址上的值。这里的一个单位是sizeof(int)个字节。

数组遍历方式二:使用数组名自增遍历数组元素,编译出错,错误如下:

error: value required as increment operand

因为数组名的指向是不可以改变的,使用自增运算符自增就会改变其指向,这是不对的,数组名只能指向数组的开头。但是可以改为如下遍历方式:

for ( i = 0; i < 5; i++ )
{printf("a[%d] = %dn", i, *(a+i));
}

这可以正确遍历数组元素。因为*(a+i)与a[i]是等价的。

2、区别二

字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。

请看如下代码:

//字符串定义方式一
char str[] = "happy";
​
//字符串定义方式二
char *str = "happy";

字符串定义方式一:字符串中的字符是可以改变的。如可以使用类似str[3]='q'这样的语句来改变其中的字符。原因就是:这种方式定义的字符串保存在全局数据区或栈区,是可读写的。

字符串定义方式二:字符串中的字符是不可以改变的。原因就是:这种方式定义的字符串保存在常量区,是不可修改的。

2、区别三

求数组长度时,借用数组名可求得数组长度,而借用指针却得不到数组长度。

请看如下代码:

#include <stdio.h>
​
int main(void)
{int a[] = {0, 1, 2, 3, 4}, *p = a;char len = 0;
​// 求数组长度方式一printf("方式一:len=%dn",sizeof(a)/sizeof(int));
​// 求数组长度方式二printf("方式二:len=%dn",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 <stdio.h>
//这是一个指针函数的声明
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] = %dn", 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 <stdio.h>
​
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 = %dn", res);
​return 0;
}
​
int add(int a, int b)
{return a + b;
}

程序运行结果如下:

以上就是关于指针函数与函数指针的简单区分。其中,函数指针广泛应用于嵌入式软件开发中,其常用的两个用途:调用函数和做函数的参数。

参考资料:

《让你不害怕指针》
C语言中文网

c char转int_C指针精华知识大汇总相关推荐

  1. 陈老师给你介绍半导体功率器件知识大汇总

    常用的半导体功率器件知识大汇总 电力电子器件(Power Electronic Device),又称为功率半导体器件,用于电能变换和电能控制电路中的大功率(通常指电流为数十至数千安,电压为数百伏以上) ...

  2. 编程精华资源大汇总 (转)

    博客是记录学习历程.分享经验的最佳平台,多年以来,各路技术大牛在ITeye网站上产生了大量优质的技术文章,并将系列文章集结成专栏,以便读者能够更便捷.更系统地浏览学习,这些可称之为"编程精华 ...

  3. Java并发编程知识大汇总

    线程简介 什么是线程 现代操作系统调度的最小单元是线程,也叫轻量级进程,在一个进程里可以创建很多是线程,这些线程都有自己的计数器,堆栈和局部变量等属性,并且能够访问共享的内存变量. 之所以我们感觉不到 ...

  4. 计算机网络基础知识大汇总

    本文对计算机网络的知识进行了全面梳理和总结,一篇文章就让你学会"计算机网络". 同时,还附带计算机网络的全套资料,可以关注公众号[小杰码匠],回复[计算机网络]免费获取. 一.什么 ...

  5. C语言、嵌入式位操作精华技巧大汇总

    最近有点忙,好久没分享笔记了~今天分享关于位操作的一点小笔记. 一.位操作简单介绍 首先,以下是按位运算符: 在嵌入式编程中,常常需要对一些寄存器进行配置,有的情况下需要改变一个字节中的某一位或者几位 ...

  6. 【基础概念大总结】— 网络安全 Java 基础知识大汇总

    这一篇,我们介绍一下:Java语言中存在有哪些安全漏洞. 我们在渗透测试中,可以如何利用这些漏洞. 那这一篇,首先全面总结一下JAVA中的基础概念. 上课了~~~上课了~~· 目录 JVM JDK J ...

  7. python产生二维复数_Python基础知识大汇总--从放弃到入门

    1.python版本选择 初学python的同学,你可能知道有python2或者python3两个不同的版本,有些较老的视频或者较旧的文档还在推崇python2,认为python2比较稳定,很多公司都 ...

  8. 2W五千字的C++基础知识整理汇总

    最近发现一篇两万五千字的C++基础知识大汇总,可把我高兴坏了.白嫖,必须白嫖! 来源:高效程序员 一 .从"hello world" 入门C++! C++总览简介 C++ 是一种静 ...

  9. c 初始化char**_CC++|指针详述及实例分析

    指针是C语言中的精华,也是一把双刃剑,关系到安全和效率. 1 系统内存布局 2 存储变量的内存地址 3 指针定义:变量,地址,类型(宽度) 4 指针声明 5 &与*运算符 6 定义指针与解引用 ...

最新文章

  1. 软件工程需求设计说明书
  2. 程序员带你解析Python3
  3. 华为200W年薪应届博士刷屏后,快手拉出了一个排的顶级名校工程师
  4. 通过SecureCRT连接Vmware中的RHEL6.3的方法
  5. 七、股票中的布朗运动和pandas.dataframe.pct_change()
  6. 课堂派派典型用户和场景
  7. VC控件 Check Box
  8. yum 出错,提示Segmentation Fault (core Dumped) 的解决办法
  9. android 元素点击位置,appium自动化操作之元素定位点击事件全家桶(find_element_by、find_elements_by)...
  10. C语言 输入的注意事项 EOF的应用
  11. 中兴路西藏北路到浦东国际机场路线(ddmap20100114)
  12. Eigen教程(9)之Reshape,Slicing介绍
  13. 457.环形数组循环
  14. Android中注册一个 BroadcastReceiver的代码
  15. 2016年 最火的 15 款 HTML5 游戏引擎
  16. SpringBoot项目从IE浏览器跳转至谷歌浏览器并打包成windows环境下可行EXE文件
  17. 程控电源测试使用小记
  18. 新概念第二册 Lesson 1 句子 结构 分析
  19. ARM 电源管理 4种模式
  20. CentOS 7 时区设置

热门文章

  1. python最基本的规则是什么_Python基础介绍(一)
  2. javascript中的事件处理
  3. FFmpeg--av_register_all函数分析
  4. 高级C语言教程-C语言函数setjmp()函数
  5. python的魔法方法--__
  6. react创建新项目
  7. 从SOURCE_BUFFER单元开始存放了20个字母A, 编程将这20个字母A的字符 串向下移10个单元.
  8. 双向循环链表的插入排序
  9. Docker搭建Nacos1.3+Seata1.4+MySQL8分布式事务(服务端)
  10. 数据结构:表达式之中缀转后缀