【C语言】“指针的运算”、“指针与数组”


一、指针运算

指针可以进行±整数,指针-指针,还有关系运算,其他的运算会被编译器阻止。

1.指针 + - 整数

对指针进行±的时候一定要注意不要越界访问,不然会产生野指针。

void print(int* arr)
{
    for (int* i = arr; i < &arr[arr_lenght];)
    {
        printf("%d", *i++);	//对指针进行++达到遍历数组的效果
    }
}

int main()
{
    int arr[arr_lenght] = { 1,2,3,4,5,6,7,8,9,0 };
    print(arr);
}

在这里插入图片描述

  • int* i = arr(数组名为首地址),此时i = 0xffee0000 对int*类型进行±整数就是对地址进行±一个int类型大小,+1之后为0xffee0004,也就是arr[1]的地址,再+1就是arr[2]的地址,依次类推,再在 i 的前面加上 * 号,就可以用指针做到遍历数组。

2.指针-指针

指针+整数为地址,所以指针-指针也是一个整数

  • 应用场景举例:我们可以在求字符串长度的时候,用字符串结尾地址-字符串首地址,得到字符串长度。
int my_strlen(char *str)		
{	
	char *pstr = str;			//先将字符串首地址进行存储
	while(*pstr != '\0' )		//strlen不应计算\0,所以遇到\0停止
    {
        pstr++;					//将字符串地址进行++
    }
	return pstr-str;			//用\0位置的地址减去首地址则为字符串长度。
}

3.指针关系运算

​ 指针还可以进行关系运算,但是标准规定:允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。

在这里插入图片描述

正确

void print(int* arr)
{
    for (int* i = arr; i < &arr[arr_lenght];)
    {
        printf("%d", *i++);	//对指针进行++达到遍历数组的效果
    }
}

int main()
{
    int arr[arr_lenght] = { 1,2,3,4,5,6,7,8,9,0 };
    print(arr);
}

错误

  • 此种办法在一些编译器下可以运行,但是我们要避免这么写!
void print(int* arr)
{
    for (int* i = &arr[arr_lenght-1]; i >= &arr[0];)
    {
        printf("%d", *i--);
    }
}

int main()
{
    int arr[arr_lenght] = { 1,2,3,4,5,6,7,8,9,0 };
    print(arr);
}

二、指针与数组

代码示例:

#include <stdio.h>
int main()
{
	int arr[10] = {1,2,3,4,5,6,7,8,9,0};
	printf("%p\n", arr);
	printf("%p\n", &arr[0]);
	return 0;
}

在这里插入图片描述

​ 我们看到数组名的地址与首元素的地址相等,那么数组名就是首元素的地址,这是在大多数情况,不过也有例外。

  • 两种特殊情况,数组名代表整个数组

(1)当sizeof(数组名)

#include <stdio.h>
int main()
{
	int arr[10] = {1,2,3,4,5,6,7,8,9,0};
	int sz = sizeof(arr);
    printf("%d",sz);			//输出40
}

数组大小为40,40就是整个数组的大小

(2)当 &数组名 时

#include <stdio.h>
int main()
{
	int arr[10] = {1,2,3,4,5,6,7,8,9,0};
	printf("%p\n", arr);
	printf("%p\n", &arr);
	printf("%d\n", arr + 1);
	printf("%d\n", &arr + 1);
}

在这里插入图片描述

我们看到&arr+1加了28(10进制的40),加的其实就是整数数组的大小

三、二级指针

​ 我们知道指针存放的其实就是变量的地址,但实质上指针也是一个变量,那么就需要一个可以用来存储指针的变量,我们称它为二级指针。(存放二级指针还可以用三级指针,存放三级指针还可以….)。

代码示例

#include <stdio.h>

int main()
{
	int a = 10;
    int *pa = &a;			//int 是指针指向变量的类型 *表示pa是个指针变量。
    int **ppa = &pa;		//int* 是指针指向变量的类型 *表示ppa是个指针。
    
    printf("%d",a);			//10
    printf("%d",*pa);		//10 要拿到一级指针的值需要解引用一次  
    printf("%d",**pa);		//10 要拿到二级指针的值需要解引用两次
}

在这里插入图片描述

  • pa存放的是a的地址,ppa存放的是pa的地址

四、指针数组

指针数组是存放指针的数组,注意要和数组指针区分开,数组指针是指存放数组的指针。

  • 数组不仅可以存储char、short、int等等的基本数据类型,还可以存储指针

#include <stdio.h>
int main()
{
    int a = 0;
    int b = 0;
    int c = 0;
	int* arr[3] = {&a,&b,&c};		//指针数组就是数组的每个位置都存储一个地址
}

完结

创作不易,还请各位小伙伴多多点赞👍关注✨收藏⭐

请添加图片描述

  • 10
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 20
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值