【C语言】指针基础知识点汇总

指针(一) 基础

一 指针的概念

  1. 为了方便访问内存中的内容,给每个内存单元一个编号,我们称这个编号为地址,及为指针.
  2. 指针也是一种数据类型 所以指针都有自己的内存 存储的是地址(编号)

指针的四要素

  1. 指针本身的类型 除去指针名,剩下的就是指针本身的类型
  2. 指针指向的类型 除去指针名和一个*,剩下的就是指针指向的类型
  3. 指针本身的内存 用来存储一个编号(4字节)
  4. 指针指向的内存 可以是各种类型
    int num = 0;          // 类型: int
    int arr[10] = {};     // 类型: int [10]
    int* MuShan = {1,2,3};
    //去掉名字,剩下的就是类型

二 指针的定义

1 运算符

*: 定义时,表示定义的是一个指针 其他的时候表示解析引用

&: 取地址符(用于取地址)

2 定义

指针的定义有多种理解方式;

  1. 类型 变量名;
int* p; // 类型: int
        // 变量名: *p
  1. 指针本身类型 指针名;
int* p; // 指针本身的类型: int*
        // 指针名: p
  1. 指针指向的类型* 指针名;
int* p; // 指针指向的类型: int
        // 指针名: p

反推:

  1. 除去指针名,剩下的就是指针本身的类型
  2. 除去指针名和一个*,剩下的就是指针指向的类型
int****** p1; // p1本身的类型: int****** 
              // p1指向的类型: int*****  (6级指针可以指向5级指针)

三 指针的内存

  1. 所有的指针,无论类型,在内存中都占4个字节的内存(存的是地址)(具体和64/32位环境相关)
#include <stdio.h>
int main()
{
	char*       pch;
	short*      psh;
	int*        pn;
	float*      pf;
	double*     pd;
    
	printf("%d\n", sizeof(pch));  // 4
	printf("%d\n", sizeof(psh));  // 4
	printf("%d\n", sizeof(pn));   // 4
	printf("%d\n", sizeof(pf));   // 4
	printf("%d\n", sizeof(pd));   // 4
    
	return 0;
}
  1. 指向起始地址
int num = 10;
int* p = &num;

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-naQJWXaU-1644340436740)(C:\Users\admin\Desktop\六星教育\学习\指针\指针(一)]\指针基础(一)1-1.png)

四 指针的初始化与赋值

1 用对应类型变量的地址

	int num = 10;
	int* pn = &num;   //初始化

	float f = 3.14f;
	float* pf;
	pf = &f;          //赋值

2 用相同类型的指针

	int num = 10;
	int* pn = &num;     // 初始值
	int* p1 = pn;       // 初始化
	int* p2;
	p2 = pn;            // 赋值

3 直接用地址

	int* p = (int*)0x36;

4 用数组名

一级指针可以接受一堆一位数组的数组名

	int arr[5] = { 1,2,3,4,5 };
	int* p = arr;

5 字符串

#include <stdio.h>
int main()
{
	// 数组名就是数组的首地址
	char arr[5] = { 'a','b','c' };
	char* p = arr;
	printf("%s\n",p);     //输出:abc
	char str[5] = "abcd";
	p = str;
	printf("%s\n", str);  //输出:abcd
	printf("%s\n", p);    //输出:abcd
	// ==> char*类型的指针可以直接用来打印整个字符串到'\0'停止
	const char* p1;
	p1 = "1234";
	printf("%s\n",p1);    //输出:1234
	const char* p2 = "Mushan";
	printf("%s\n",p2);    //输出:Mushan
	return 0;
}

6 置空

	int* p = NULL;
	int* p1 = (int*)0x0;
/*
   NULL: #define NULL  0
   0地址
   有时候,指针定义好了,但是暂时没有指向
   或者是,指针用完了,没有指向了
   指针不知道指向哪里(会随机指向)
   此时的指针,很危险(野指针)
   所以 这些情况下的指针
   统一安排一个地址给他们指向
   指向0地址
*/

7 多级指针

#include<stdio.h>

int main() {
	int num = 10;
	printf(" num = %d\n", num);        //   num = 10;
	printf("&num = %X\n", &num);       //  &num = 10FFA78
    
	int* p = &num;
	printf("*p = %d\n", *p);           //  *p = 10 (num值)
	printf(" p = %X\n", p);            //   p = 10FFA78 (num的地址)
	printf("&p = %X\n", &p);           //  &p = 10FFA6C

	int** pp = &p; // 一个二级指针
	printf("**pp = %d\n", **pp);       // **pp = 10 (num值)
	printf(" *pp = %X\n", *pp);        //  *pp = 10FFA78 (num的地址)
	printf(" pp = %X\n", pp);          //   pp = 10FFA6C (p的地址)
	printf(" &pp = %X\n", &pp);        //  &pp = 10FFA60 

	int*** ppp = &pp; // 一个三级指针
	printf("***ppp = %d\n", ***ppp);   // ***ppp = 10 (num值)
	printf(" **ppp = %X\n", **ppp);    //  **ppp = 10FFA78 (num地址)
	printf(" *ppp = %X\n", *ppp);      //   *ppp = 10FFA6C (p的地址)
	printf("  ppp = %X\n", ppp);       //    ppp = 10FFA60 (pp的地址)
	printf(" &ppp = %X\n", &ppp);      //   &ppp = 10FFA54

	return 0;
}

五 指针的加减法

核心:指针本身的值(指向)没有变化

指针偏移
  1. 指针可以加上或减去一个整数

  2. 指针加上或减去一个整数后,实际上是进行了偏移

  3. 偏移的范围是加上或减去的整数个单位

    单位: 指针指向的类型在内存中所占字节数

    偏移:指针指向不变,但是可以根据偏移量取内容

#include <stdio.h>
int main()
{
	int num = 10;
    
	int* p = &num;
	printf("%X\n", p);          // EFFB5C
	printf("%X\n", p + 1);      // EFFB60
    
	return 0;
}

六 指针的自增自减

自增自减,会改变指针指向

++:表示指针向后移动一个单位

– :表示指针向前移动一个单位

单位:指针指向的类型在内存所占内存中所占的字节数

#include <stdio.h>
int main()
{
	int num = 10;
	int* p = &num;
	printf("%X\n", p);          // EFFB5C
	printf("%X\n", p + 1);      // EFFB60

	printf("%d\n",*p);          // 10
	printf("%X\n", p += 1);     // EFFB60
	printf("%d\n",*p);          // -858993460(无意义)
	return 0;
}

七 通过指针遍历数组

遍历一维数组

#include <stdio.h>
int main()
{
	int arr[10] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
	for (size_t i = 0; i < 10; i++)
	{
		printf("%2d", arr[i]);
	}
	printf("\n");
	int* p = arr;
	// p和arr,除了arr是一个常量之外,其他几乎是一样的
	for (size_t i = 0; i < 10; i++)
	{
		printf("%2d", p[i]);
	}

	printf("\n");
	printf("%d\n", p[0]);        // 0
	printf("%d\n", *(p + 0));    // 0
	printf("%d\n", p[1]);        // 1(先偏移 后取值)
	printf("%d\n", *(p + 1));    // 1
	// p[n] <==> *(p+n)
	return 0;
}
// p[n]:叫做下标形式
// *(p+n):叫做指针偏移的形式

遍历二维数组

  1. 二维数组也是数组

  2. 二维数组可以看成元素是一维数组的一维数组

  3. 数组的内存是连续的

#include<stdio.h>

int main()
{
	int arr[3][4] = {
        {1,2,3,4},
		{5,6,7,8},
		{9,10,11,12}
	};

	for (size_t i = 0; i < 3; i++)
	{
		for (size_t j = 0; j < 4; j++)
		{
			printf("%3d", arr[i][j]);
		}
		printf("\n");
	}

	int* p0 = arr[0];
	int* p1 = arr[1];
	int* p2 = arr[2];
	printf("\n");
	// 1:
	for (size_t i = 0; i < 4; i++)
	{
		printf("%3d", p0[i]);           // 1  2  3  4
	}
	printf("\n");
	for (int i = -4; i <= 7; i++)
	{
		printf("%3d", p1[i]);           // 1  2  3  4  5  6  7  8  9 10 11 12
	}
	printf("\n");
	for (int i = 0; i < 12; i++)
	{
		printf("%3d", arr[0][i]);       // 1  2  3  4  5  6  7  8  9 10 11 12
	}
	printf("\n");
	// 下标: 保证数组不越界即可
    
	// 2:
	int* p = &arr[0][0];
	for (int i = 0; i < 12; i++)
	{
		printf("%3d", arr[0][i]);       // 1  2  3  4  5  6  7  8  9 10 11 12       
	}
	printf("\n");
	for (int i = 0; i < 12; i++)
	{
		printf("%3d", *p);              // 1  2  3  4  5  6  7  8  9 10 11 12
		p++;
	}
	printf("\n");
	return 0;
}

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

MuShan-bit

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

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

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

打赏作者

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

抵扣说明:

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

余额充值