C语言之指针高级--指针操作二维整型、字符型数组、函数指针

指针操作二维整型数组

回顾:在C语言中并不存在真正的二维数组,二维数组的本质 是一维数组的一维数组 ,比如定义一个二维数组 int a[2][3],本质上是 int [3] a[2],表示定义了一个数组类型(具体类型是int [3])的一维数组a[2],在学习指针操作二维数组时,要从其本质出发,否则很可能被绕晕。二维数组 也 符合数组的特点 ,即连续性,有序性,单一性 。

指针定义的语法是

基类型 * 变量名

那么应该需要确定一个基类型是什么的指针,来指向二维数组呢?

如果对二维数组本质比较清楚,那么a[2][3]中,a[0](即所谓的"第一行")的类型是int [3]这种数组类型,由此可以推断 &a[0] 对应就是 int [3]*,只不过C语言中不允许这种写法,所以正确的写法是int (*) [3],即定义了一个数组类型的指针,就是数组指针。所以我们就可以定义一个这样类型的指针:

int (* p)[3] = a // 表示p指向了二维数组a,p的基类型相当于int [3]

*p <=> a[0] //相当于是内部的一维数祖名

(*p)[0] <=> *(*p+0) <=> **p // 都表示a[0][0];

*(*(p+1)+1) <=> a[1][1]

*(*(p+i)+j) <=> a[i][j]

若a是二维数组,则a[i] 代表一维数组名,它只是一个地址,并不代表某一个元素的值,所以a 、a+i、a[i]、*(a+i)、*(a+i)+j、a[i] + j都是地址;而*(a[i] + j)和 *(* (a+i) + j) 是二维数组元素a[i] [j]的值。

注意:

二维数组的操作

从二维数组的本质 进行的 ,直接的访问操作也是 一维一维的展开 。

以下是简单使用实例,实现二维数组打印

#include <stdio.h>

int main(void)
{

	int a[2][3] = {1,2,3,4,5,6};

	int (*p)[3] = a;
	
	int i = 0;
	int j = 0;
	for (i = 0;i < 2; ++i)
	{
		for (j = 0; j < 3; ++j)
		{
			//相当于printf("%d ",a[i][j]);
			printf("%d ",*(*(p+i) + j));
		}
		putchar('\n');
	}
	
	return 0;
}

指针操作二维字符型数组

二维字符型数组主要用来处理字符串,所以指针操作二维字符型数组,就是操作字符串。定义指针变量类似二维整型数组。

但是有两种方式可以给值:

方式1

char s[ ] = {"hello", "world"};

char (*p) [10] = s; //表示p指向二维数组s

因为二维字符型数组主要操作字符串,所以我们是取一整个字符来进行操作的,所以用指针访问时,直接 *(p+i)就可以取到字符串,这是区别于二维字符型数组的地方。

方式2

在一维字符型数组中,我们可以 char * p = "hello",表示定义了一个指针变量,直接指向了字符串,就是p存了字符串常量区中放hello的地址。那么对于多个存放字符串的地址,我们可以:

char * p1 = "hello";

char * p2 = "world";

char * p3 = "China";

......

C语言中用数组来批量处理数据,所以定义了一个数组:

char * p[3] = {"hello", "world","China"};

因为要存放地址这种类型,所以基类型是char *,也就是指针类型,数组p里面存放的是指针类型的变量----地址。

对于p[3]这个数组来说,里面存放的是各个字符串的地址,也就是存放着地址数据的数组,就是**指针数组,**注意区别于数组指针。

所以如果要定义一个指针来指向这个数组,可以写成:

char * * q = p; //q是二级指针

char * 代表基类型是指针类型

* 代表定义了一个指针类型的变量q,只是一个声明,无其它含义。

应用:

int main (int argc, const char * argv[ ])

在C语言和C++语言中,main 函数是程序的入口点,它可以接受两个参数: argc 和 argv[] 。这两个参数允许程序接收从命令行传递给程序的参数。

  1. **argc**:这是一个整型变量,表示命令行中传递给程序的参数个数。它包括程序本身的名称,所以至少为1。例如,如果在命令行中运行 program arg1 arg2,argc 的值将是3。

  2. **argv[]**:这是一个字符指针数组,每个元素指向一个以null结尾的字符串。argv[0]是程序本身的名称,argv[1]是第一个参数,argv[2]`是第二个参数,依此类推。argv数组的最后一个元素是一个空指针(即NULL或`nullptr),表示参数列表的结束。

用法示例:

假设有一个C程序,其main函数定义如下:

int main(int argc, char *argv[]) {

// 程序代码

return 0;

}

如果在命令行中运行这个程序并传递参数,如下

./myprogram arg1 arg2 arg3

那么在程序中,argc的值将是4,argv数组将包含以下元素:

argv[0] -> "./myprogram"

argv[1] -> "arg1"

argv[2] -> "arg2"

argv[3] -> "arg3"

argv[4] -> NULL

在程序中,可以使用这些参数来执行不同的操作,例如

int main(int argc, char *argv[]) {

if (argc > 1) {

printf("Number of arguments: %d\n", argc - 1);

for (int i = 1; i < argc; i++) {

printf("Argument %d: %s\n", i, argv[i]);

}

}

return 0;

}

这个程序将打印出除了程序名称之外的参数个数,以及每个参数的值。

指针 + 函数

通过指针 的方式 来调用函数

函数名 --- 代表函数的入口地址

int add(int a,int b) // 求和函数

// 函数名 - 代表函数的入口地址

// 函数名对应的数据类型

// int (int a,int b) //函数类型

// 代表一类函数

// 返回值为int型

// 带有两个int型的形参变量

说明:

1.可以定义一个函数类型的指针变量

来保存函数的入口地址

int (*p)(int, int ) = add

2.有了这个指针变量

通过指针变量 进行函数调用

int ret = p(1,2)

一般用于实现函数回调(callback)

回调函数 --- 通过函数指针调用函数 就是回调。c语言中,使用了函数指针实现回调。

以下是回调的简单实例:

#include <stdio.h>

int add(int a,int b)
{
	return a + b;
}

int sub(int a,int b)
{
	return a - b;
}
int mul(int a,int b)
{
	return a * b;
}
int div(int a,int b)
{
	return a / b;
}


void processData(int a,int b,int(*p)(int,int))
{
	printf("result: %d\n",p(a,b));
}

int main(int argc, const char *argv[])
{

	processData(1,2,add);
	processData(1,2,sub);
	processData(1,2,mul);
	processData(1,2,div);
	
	return 0;
}
相关推荐
阿俊仔(摸鱼版)6 分钟前
Python 常用运维模块之OS模块篇
运维·开发语言·python·云服务器
军训猫猫头6 分钟前
56.命令绑定 C#例子 WPF例子
开发语言·c#·wpf
sunly_13 分钟前
Flutter:自定义Tab切换,订单列表页tab,tab吸顶
开发语言·javascript·flutter
远方 hi24 分钟前
linux虚拟机连接不上Xshell
开发语言·php·apache
涅槃寂雨24 分钟前
C语言小任务——寻找水仙花数
c语言·数据结构·算法
『往事』&白驹过隙;31 分钟前
操作系统(Linux Kernel 0.11&Linux Kernel 0.12)解读整理——内核初始化(main & init)之缓冲区的管理
linux·c语言·数据结构·物联网·操作系统
就爱学编程32 分钟前
从C语言看数据结构和算法:复杂度决定性能
c语言·数据结构·算法
涛ing33 分钟前
23. C语言 文件操作详解
java·linux·c语言·开发语言·c++·vscode·vim
NoneCoder34 分钟前
JavaScript系列(42)--路由系统实现详解
开发语言·javascript·网络
半桔37 分钟前
栈和队列(C语言)
c语言·开发语言·数据结构·c++·git