C语言指针—指针和指针类型


  1. 指针是什么?

    指针是什么?

    指针理解的2个要点:

    指针是内存中一个最小单元的编号,也就是地址

    平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量

    总结:指针就是地址,口语中说的指针通常指的是指针变量。

    那我们就可以这样理解:

    内存

指针变量

我们可以通过&(取地址操作符)取出变量的内存其实地址,把地址可以存放到一个变量中,这个 变量就是指针变量

#include <stdio.h>
int main()
{
 int a = 10;//在内存中开辟一块空间
 int *p = &a;//这里我们对变量a,取出它的地址,可以使用&操作符。
    //a变量占用4个字节的空间,这里是将a的4个字节的第一个字节的地址存放在p变量
中,p就是一个之指针变量。
 return 0;
}

总结:

指针变量,用来存放地址的变量。(存放在指针中的值都被当成地址处理)。

那这里的问题是:

  • 一个小的单元到底是多大?

  • (1个字节) 如何编址?

经过仔细的计算和权衡我们发现一个字节给一个对应的地址是比较合适的。

对于32位的机器,假设有32根地址线,那么假设每根地址线在寻址的时候产生高电平(高电压)和低电 平(低电压)就是(1或者0); 那么32根地址线产生的地址就会是:

00000000 00000000 00000000 00000000

00000000 00000000 00000000 00000001

11111111 11111111 11111111 11111111

这里就有2的32次方个地址。

每个地址标识一个字节,那我们就可以给 (2^32Byte == 2^32/1024KB == 232/1024/1024MB==232/1024/1024/1024GB == 4GB) 4G的空闲进行编址。

同样的方法,那64位机器,如果给64根地址线,那能编址多大空间,自己计算。

这里我们就明白:

  • 在32位的机器上,地址是32个0或者1组成二进制序列,那地址就得用4个字节的空间来存储,所以 一个指针变量的大小就应该是4个字节。
  • 那如果在64位机器上,如果有64个地址线,那一个指针变量的大小是8个字节,才能存放一个地 址。

总结: 指针是用来存放地址的,地址是唯一标示一块地址空间的。 指针的大小在32位平台是4个字节,在64位平台是8个字节。

指针和指针类型

指针也是有类型的

X86  32位环境    
X64  64位环境    
int main()
{
    char* pc = NULL;
    short* ps = NULL;
    int* pi = NULL;
    double* pd = NULL;

    //sizeof返回的值是无符号整型 unsigned int    用%zu来打印
    printf("%zu\n", sizeof(pc));
    printf("%zu\n", sizeof(ps));
    printf("%zu\n", sizeof(pi));
    printf("%zu\n", sizeof(pd));
    return 0;
}

现在用64位环境运行上面的代码

image-20240408173638195

现在用86位环境运行上面的代码

image-20240408173708270

既然指针大小都是固定的,为什么还要区分指针类型呢?


指针类型

我们来看下面的代码

int main()
{
    int a = 0x11223344;//0x开头的是16进制数字,两个十六进制数字代表一个字节
    //0001 0001 0010 0010 0011 0011 0100 0100

    return 0;
}

我们看调试窗口

image-20240408174822352

image-20240408174925194

可以看见确实是占用了四个字节

int main()
{
    int a = 0x11223344;//0x开头的是16进制数字,两个十六进制数字代表一个字节
    //0001 0001 0010 0010 0011 0011 0100 0100
    int* pa = &a;
    *pa = 0;
    return 0;
}

image-20240408175129774

我们再往下一步可以看到四个字节一下子都被修改了

image-20240408175216818

那如果我们改为char*呢?

image-20240408175433672

我们只改动了一个字节

因为我们所使用的是char*类型的指针,所以一次只能修改一个字节

指针类型的差异并没有决定所占大小,但是决定了被解引用所访问的字节


指针±整数

int main()
{
	int n = 10;
	char* pc = (char*)&n;
	int* pi = &n;

	printf("%p\n", &n);
	printf("%p\n", pc);
	printf("%p\n", pc + 1);
	printf("%p\n", pi);
	printf("%p\n", pi + 1);
	return  0;
}

image-20240408180705110

我们可以看到,两个地址放的是同一个值

但是int*往前走了4个字节

char*只走了1个字节

**结论:**指针的类型决定了指针+1操作时跳过几个字节

可以通俗的理解为决定了指针的步

int和float都是解引用访问四个字节,那么int*和float*可以通用吗

int main()
{
	int a = 0;

	int* pi = &a;
	float* pf = &a;

	*pi = 100;

	return  0;
}

可以看到已经成功存入

image-20240408181754016

那如果是这样呢

int main()
{
	int a = 0;

	int* pi = &a;
	float* pf = &a;

	*pf = 100.0;

	return  0;
}

image-20240408181928789

那这样子就不一样了,float*认为内存中放的就是浮点数,所以放进内存的时候默认为浮点数形式

那么照这么看,int*和float*就是不能混用的

  • 20
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

J.Pei

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值