当前位置: 首页 > news >正文

10.1 指针进阶

指针进阶

  • 1. 字符指针
  • 2. 指针数组
  • 3. 数组指针
    • 3.1 定义
    • 3.2 数组名; &数组名[]; &数组名
    • 3.3 数组指针的使用
  • 4. 数组传参和指针传参
    • 4.1 一维数组传参
    • 4.2 二维数组传参
    • 4.3 一级指针传参
    • 4.4 二级指针传参
  • 5. 函数指针

1. 字符指针

int main()  
{  char ch = 'w';  char *pc = &ch;//pc就是字符指针*pc = 'a';  return 0;  
}
int main()
{char arr[] = "abcdef";//[a b c d e f \0]const char* p = "abcdef";//常量字符串//把首字符a的地址赋值给p//p指向a的地址printf("%s\n", p);printf("%c\n", *p);return 0;
}
abcdef
a

分析代码

int main()
{char str1[] = "hello RuaRua.";char str2[] = "hello RuaRua.";const char* str3 = "hello RuaRua.";const char* str4 = "hello RuaRua.";if (str1 == str2)printf("str1 and str2 are same\n");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("str3 and str4 are not same\n");return 0;
}
str1 and str2 are not same
str3 and str4 are same
  • C/C++会把常量字符串存储到单独的一个内存区域,str3str4 是指向常量字符串的指针,它们指向的是存储在静态存储区只读部分的字符串常量 "hello RuaRua.",这个字符串常量与 str1str2 中存储的字符串虽然内容相同,但在内存中是不同的存储位置。

2. 指针数组

int* arr1[10]; //存放  整形指针  的数组  
char *arr2[10]; //存放  一级字符指针  的数组  
char **arr3[10];//存放  二级字符指针  的数组

eg.模拟二维数组

int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 2,3,4,5,6 };int arr3[] = { 3,4,5,6,7 };int* arr[3] = { arr1,arr2,arr3 };int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 5; j++){printf("%d ", arr[i][j]);}printf("\n");}return 0;
}
1 2 3 4 5
2 3 4 5 6
3 4 5 6 7

3. 数组指针

3.1 定义

int (*p)[10];
//p是数组指针变量,指向的是数组

3.2 数组名; &数组名[]; &数组名

int main()
{int arr[10] = { 0 };printf("%p\n", arr);printf("%p\n", &arr[0]);printf("%p\n", &arr);printf("%d\n", sizeof(arr));return 0;
}
008ff858    arr     是数组首元素地址
008ff858    &arr[0] 是数组首元素地址
008ff858    &arr    是整个数组地址
40          sizeof(arr)   计算整个数组大小,arr表示整个数组
名称类型
arr0x008ff858 {0, 0, 0, 0, 0, 0, 0, 0, 0, 0}int[10]
&arr[0]0x008ff858 {0}int *
&arr0x008ff858 {0, 0, 0, 0, 0, 0, 0, 0, 0, 0}int(*)[10]
int main()
{int arr[10] = { 0 };printf("%p\n", arr);printf("%p\n", arr+1);printf("%p\n", &arr[0]);printf("%p\n", &arr[0]+1);printf("%p\n", &arr);printf("%p\n", &arr+1);return 0;
}
00F3F840
00F3F844
00F3F840
00F3F844
00F3F840
00F3F868-->0x28-->40
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };//把数组arr的地址赋值给数组指针变量p //int[10] * p = &arr;//errint(*p)[10] = &arr;//p是数组指针,存放数组地址return 0;
}
arr   的类型是 int [10]
p     的类型是 int(*)[10]
&arr  的类型是 int(*)[10]

3.3 数组指针的使用

一般在二维数组中使用,常规写法:

void Print(int arr[3][5], int r, int c)//row,column
{int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 5; j++){printf("%d ", arr[i][j]);}printf("\n");}
}
int main()
{int arr[3][5] = { 1,2,3,4,5, 2,3,4,5,6, 3,4,5,6,7 };Print(arr, 3, 5);return 0;
}

二维数组,每一行可以理解为数组的一个元素
每一行是一个一维数组
二维数组是 一维数组的数组

arr -->首元素地址-->第一行地址-->一维数组地址
void Print(int(*p)[5], int r, int c)
{int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 5; j++){printf("%d ", *(*(p + i) + j));}printf("\n");}
}
(p + i):指针移动到i数组
*(p + i):i数组的地址
(*(p + i) + j):i数组j元素的地址
*(*(p + i) + j):得到元素

4. 数组传参和指针传参

4.1 一维数组传参

void test1(int arr[5], sz)
{}
void test2(int* p, sz)
{}
int main()
{int arr[5] = { 0 };test1(arr, 5);test2(arr, 5);return 0;
}

4.2 二维数组传参

void test1(int arr[3][5], int r, int c)
{}
void test2(int(*p)[5], int r, int c)
{}
int main()
{int arr[3][5] = {0};test1(arr, 3, 5);test2(arr, 3, 5);return 0;
}

int arr[3][5]:是二维数组类型的形参,虽然写成 int arr[3][5] ,但实际上会退化为指向包含 5int 类型元素的数组的指针,即 int (*)[5] 类型。
int (*p)[5]:是指针类型的形参,p 是一个指向包含 5int 类型元素的数组的指针。和 int arr[3][5] 本质上是等价的,都用于接收二维数组的首地址,即第一行的地址。

4.3 一级指针传参

void print(int* p, int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d\n", *(p + i));}
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9 };int* p = arr;int sz = sizeof(arr) / sizeof(arr[0]);//一级指针p,传给函数print(p, sz);return 0;
}

当函数的参数为一级指针,函数能接收什么参数?

void test(int* p)//函数能接收什么参数?
{ }
int n = 0;
int *ptr = &n;
int arr[] = "abcdef";
test (ptr);
test (&n);
test (arr);

4.4 二级指针传参

void test(int** ptr)
{printf("num = %d\n", **ptr);
}
int main()
{int n = 10;int* p = &n;int** pp = &p;test(pp);test(&p);return 0;
}

当函数的参数为二级指针的时候,可以接收什么参数?

void test(int** p)
{ }
int main()
{int n = 10;int* p = &n;int** pp = &p;int* arr[10];test(&p);test(pp);test(arr);return 0;
}

5. 函数指针

void test()
{printf("hehe\n");
}
int main()
{printf("%p\n", test);printf("%p\n", &test);return 0;
}
006013B6
006013B6
test和&test都是函数地址

eg

void test(char* pc, int arr[10])
{ }
int main()
{void (*pf)(char*, int arr[]) = test;return 0;
}

eg

int Add(int x, int y)
{return x + y;
}
int main()
{int arr[10] = { 0 };int(*pa)[10] = &arr;//类比int(*pf)(int, int) = &Add;//pf是函数指针变量return 0;
}

使用指针调用函数

int Add(int x, int y)
{return x + y;
}
int main()
{int(*pf)(int, int) = &Add;int r = Add(3, 5);printf("%d\n", r);int m = (*pf)(4, 5);//--> int m = pf(4, 5);//pf和Add一回事printf("%d\n", m);return 0;
}

eg1.代码分析

(*(void (*)())0)();void (*p)()-->p是函数指针
void (*)() -->函数指针类型
(void (*)())0-->强制类型转换, 0转换成 void (*)()0的地址中存放着这个函数
*(void (*)())0-->解引用,调用这个函数
(*(void (*)())0)()-->调用时无参数
(*(void (*)())0)();-->调用0地址处的函数

eg2.代码分析

void (*signal(int , void(*)(int)))(int);void (*p)(int); -->函数指针
void(*)(int)-->函数指针指向一个int类型的参数,返回类型void
signal(int , void(*)(int))--> signal 函数(类型 int ,函数指针类型 void(*)(int))

简化

//typedef重定义
typedef unsigned int uint;
typedef int* ptr_t;typedef int(*parr_t)[10];
typedef int(*pf_t)(int,int);int main()
{uint u1;ptr_t p1;int* p2;return 0;
}
void (*signal(int , void(*)(int)))(int);
--->typedef void(*pf_yt)(int);//void (*signal(int, void(*)(int)))(int);pf_yt signal(int, pf_yt);

后续内容:10.2 指针进阶


http://www.mrgr.cn/news/92798.html

相关文章:

  • 施耐德 PLC 硬件库 DTM 的安装
  • Mysql100道高频面试题
  • 【Qt QML】QML鼠标事件(MouseArea)
  • C#-委托
  • Spring Boot3+Vue3极速整合: 10分钟搭建DeepSeek AI对话系统(进阶)
  • SOLID Principle基础入门
  • QT基础十、表格组件:QTableWidget
  • vue3中展示markdown格式文章的三种形式
  • 前端实现OSS上传图片(Vue3+vant)
  • 14. LangChain项目实战1——基于公司制度RAG回答机器人
  • P8720 [蓝桥杯 2020 省 B2] 平面切分--set、pair
  • Metal学习笔记九:光照基础
  • latex 环境配置
  • 【系统稳定性】1.11 QVM稳定性问题分析(一)
  • 【星云 Orbit-F4 开发板】05. NVIC中断分组与配置(重要)
  • (上)基于机器学习的图像识别——遥感图像分类(LeNet-5;AlexNet;VGGNet;GoogLeNet;ResNet)
  • 如何获取mac os 安装盘
  • 线代[9]|线性代数主要内容及其发展简史(任广千《线性代数的几何意义》的附录1)
  • RocketMQ的运行架构
  • 面向AI 的前端发展及初识大模型