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

C 指针数组函数之间的关联

可能经常会听到:指针常量、常量指针、指针数组、数组指针、指针函数、函数指针;函数指针数组,等这些听起来感觉向绕口令似的词汇;
可见数组、指针、函数之间是有很多联系的。比如看下面一段代码:

#include<stdio.h>
#include <string.h>int main() {char* str = "abc";char* str1 = "12";strcpy(str, str1);return 0;
}

运行起来怎么还报错了呢?如果你回答不上来,那说明弄清上面这些名词后面的细节是十分有必要的!今天本文就带领读者回顾一下这些名词背后的知识点!

指针常量、常量指针

指针常量和常量指针,都是指针;

指针常量:

形如:const char * str,意思是告诉使用者,指针指向了常量,请你不要试图通过指针修改所指向的变量;这种常用在函数参数中,防止在函数内部不小心修改了指针指向的变量的值。

常量指针:

形如:char * const str,意思是告诉使用者,指针被声明为常量,不要试图修改这个指针的值;即:不要试图修改指针指向的地址;数组名貌似就有这种性质,声明数组后,不可对数组再次进行初始化。

如何区分指针常量和常量指针

看const和的位置,const在的左边--常量指针;const在*的右边--指针常量。

指针数组、数组指针

#include <stdio.h>
#include <string.h>int main(void) {int arr[] = {1, 2, 3};int (*p_arr)[] = &arr;//数组指针printf("%d\n", (*p_arr)[1]);char *s1[] = {"hello", "world", "!"};//指针数组printf("%s\n", s1[2]);return 0;
}

image

指针数组

看后面两个字,重在“数组”;指针数组,是说有一个数组,它是由指针类型元素组成。

数组指针

看后面两个字,重在“指针”;数组指针,是说有一个指针,它指向了一个数组。

指针函数、函数指针

指针函数

看后面两个字,重在“函数”;指针函数,是说有一个函数返回值类型是指针类型。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int *mock(int size) {return malloc(sizeof(int) * size);
}int main(void) {int *p = mock(5);p[0] = 1;p[1] = 3;p[2] = 5;p[3] = 7;p[3] = 7;free(p);int number = p[2];return 0;
}

看一下执行情况:
image
image

函数指针

看后面两个字,重在“指针”;函数指针,是说有一个指针变量,它指向一个函数。

#include <stdio.h>
#include <string.h>int add(int a, int b) {return a + b;
}int main(void) {int (*p_add)(int, int);//函数指针p_add = &add;//将函数add的地址赋值给函数指针int ret = (*p_add)(1, 2);return 0;
}

看add函数在内存中是什么类型:
image

实用技巧

常量指针做函数形参

为防止在函数内部,通过指针修改函数外部变量的值;函数参数通常定义为常量指针。
image

数组做函数形参

请看下面的函数形式:

#include <stdio.h>void test1(int a[10]) {printf("in test1 size is :%lu\n", sizeof(a));
}void test2(int a[]) {printf("in test2 size is :%lu\n", sizeof(a));
}void test3(int *a) {printf("in test3 size is :%lu\n", sizeof(a));
}int main(void) {int arr[10] = {0};test1(arr);test1(arr);test1(arr);return 0;
}

输出

in test1 size is :8
in test1 size is :8
in test1 size is :8

由于数组做函数形参会退化成指针,而对指针求sizeof得到的是指针类型占用的空间;并不能得到数组占用的空间;所以数组做函数形参时,通常还需再定义一个参数:数组的长度。
void test4(int arr[], int n);

函数指针做函数形参

使用函数指针做函数形参,可以实现函数回调的效果;如下面示例,便演示了一个简单的发布、订阅机制。

#include <stdio.h>
#include <strings.h>typedef void (*func)(void);int _index;func func_arr[10];void add_handler(void (*callback)()) {if (callback != NULL && _index < 10) {func_arr[_index++] = callback;}
}void raise_event() {int n = 0;for (; n < _index; n++) {if (func_arr[n] != NULL) {func_arr[n]();}}
}void test1() {printf("test1...\n");
}void test2() {printf("test2...\n");
}int main(void) {add_handler(test1);add_handler(test2);raise_event();return 0;
}

效果:

test1...
test2...

以上便是数组、指针和函数之间的关联的介绍。

http://www.rkmt.cn/news/47962.html

相关文章:

  • 逻辑回归(随笔)
  • 这封邮件写得真好,是你自己写的吗? 不,是AI写的
  • 灵活用工-连续劳务-计算器工具类,拿走不谢
  • Scapy构建telnet包
  • 逻辑回归原理与案例分析
  • 杂题记录 4
  • 25年11月计数题做题记录
  • CCPC2025哈尔滨站-H. 匹配
  • 【做题记录】HZOJ 多校-数论
  • 2014 吉林省赛题解 | CCUT应用OJ题解——F[X] + X = N
  • 洛谷 P4859 已经没有什么好害怕的了 题解(DP,二项式反演)
  • 飞鱼uu单人防空4
  • HaluMem:揭示当前AI记忆系统的系统性缺陷,系统失效率超50%
  • 团队作业2-需求规格说明书
  • 25.11.12 差分约束算法
  • 11/12
  • 解决Cursor编辑器无法通过include path识别C++头文件的问题
  • 重组蛋白基础与技术概述
  • Dynamics 365 Field Service跨站脚本欺骗漏洞分析
  • 日报11.12
  • [译] 省略 Async 与 Await
  • iverilog、gtkwave工具链接
  • 简化Python数据结构初始化:从繁琐到优雅的进阶指南 - 详解
  • 软工团队作业2--需求规格说明书
  • #题解#洛谷P1314#二分#前缀和#
  • 《团队作业2》需求规格说明书
  • 深入理解C++智能指针:掌握RAII与内存安全的利器 - 详解
  • Linux下的花式「隔空」文件传输魔法
  • OpenEuler 22.03 安装zabbix-agent(源代码编译及自制rpm包)
  • pq使用体验和改进建议