【C语言】函数进阶

【C语言】函数进阶


文章目录


前言

本篇文章我们会讲到有关函数的相关知识,函数指针的定义方式,函数指针数组,函数指针做函数参数,回调函数,递归函数。


一、函数指针的定义方式

  • 函数指针 和 指针函数 区别?
    函数指针 指向了函数的指针
    指针函数 函数返回值是指针的函数
c 复制代码
void func(int a, char c)
{
	printf("hello world\n");
}

void test01()
{
	//1、先定义出函数类型,在通过类型定义函数指针
	typedef void(FUNC_TYPE)(int, char);

	FUNC_TYPE* pFunc = func;
	pFunc(10, 'a');

	//2、定义出函数指针类型,通过类型定义函数指针变量
	typedef void(*FUNC_TYPE2)(int, char);
	FUNC_TYPE2 pFunc2 = func;
	pFunc2(20, 'b');

	//3、直接定义函数指针变量
	void(*pFunc3)(int, char) = func;
	pFunc3(30, 'c');

}

二、函数指针数组

c 复制代码
//函数指针的数组
void func1()
{
	printf("func1 调用了\n");
}

void func2()
{
	printf("func2 调用了\n");
}

void func3()
{
	printf("func3 调用了\n");
}

void test02()
{
	void(*pArray[3])();

	pArray[0] = func1;
	pArray[1] = func2;
	pArray[2] = func3;

	for (int i = 0; i < 3; i++)
	{
		pArray[i]();
	}
}

三、函数指针做函数参数

提供一个打印函数,可以打印任意类型的数据

c 复制代码
//提供一个打印函数,可以打印任意类型的数据
void printText(void* data, void(*myPrint)(void*))
{
	myPrint(data);
}

void myPrintInt(void* data)
{
	int* num = data;
	printf("%d\n", *num);
}



void test01()
{
	int a = 10;
	printText(&a, myPrintInt); 
}


struct Person
{
	char name[64];
	int age;
};


void myPrintPerson(void* data)
{
	struct Person* p = data;
	printf("姓名: %s 年龄:%d\n", p->name, p->age);
}


void test02()
{
	struct Person p = { "Tom",18 };
	printText(&p, myPrintPerson);

}

四、回调函数

提供一个函数,实现可以打印任意类型的数组

c 复制代码
//提供一个函数,实现可以打印任意类型的数组 
void printAllArray(void* pArray, int eleSize, int len, void(*myPrint)(void*))
{
	char* p = pArray;

	for (int i = 0; i < len; i++)
	{
		//获取数组中每个元素的首地址
		char* eleAddr = p + eleSize * i;
		//printf("%d\n", *(int *)eleAddr);
		//交还给用户做打印操作
		myPrint(eleAddr);
	}

}

void myPrintInt(void* data)
{
	int* num = data;
	printf("%d\n", *num); 
}

void test01()
{
	int arr[5] = { 1, 2, 3, 4, 5 };
	int len = sizeof(arr) / sizeof(int);
	printAllArray(arr, sizeof(int), len, myPrintInt);
}


struct Person
{
	char name[64];
	int age;
};

void myPrintperson(void* data)
{
	struct Person* p = data;
	printf("姓名:%s  年龄:%d \n", p->name, p->age);
}

//查找数组中的元素是否存在
//参数1  数组首地址   参数2  每个元素的大小  参数3  数组元素个数  参数4 查找数据
int findArrayEle(void* pArray, int eleSize, int len, void* data, int(*myCompare)(void*, void*))
{
	char* p = pArray;

	for (int i = 0; i < len; i++)
	{
		//每个元素的首地址
		char* eleAddr = p + eleSize * i;

		//if ( 数组中的变量的元素 == 用户传入的元素)
		if (myCompare(eleAddr, data))
		{
			return 1;
		}
	}

	return 0;

}
int myComparePerson(void* data1, void* data2)
{
	struct Person* p1 = data1;
	struct Person* p2 = data2;

	//if ( strcmp( p1->name , p2->name) == 0  &&  p1->age == p2->age)
	//{
	//	return 1;
	//}
	//return  0;

	return   strcmp(p1->name, p2->name) == 0 && p1->age == p2->age;

}



void test02()
{
	struct Person personArray[] =
	{
		{ "aaa", 10 },
		{ "bbb", 20 },
		{ "ccc", 30 },
		{ "ddd", 40 },
	};
	int len = sizeof(personArray) / sizeof(struct Person);
	printAllArray(personArray, sizeof(struct Person), len, myPrintperson);

	//查找数组中指定的元素是否存在
	struct Person p = { "ccc", 30 };

	int ret = findArrayEle(personArray, sizeof(struct Person), len, &p, myComparePerson);

	if (ret)
	{
		printf("找到了元素\n");
	}
	else
	{
		printf("未找到\n");
	}
}

五、递归函数

利用递归实现字符串逆序遍历
斐波那契数列

c 复制代码
//利用递归实现字符串逆序遍历
void reversePrint(char* p)
{
	if (*p == '\0')
	{
		return;
	}

	reversePrint(p + 1);

	printf("%c\n", *p);

}
void test01()
{
	char* str = "abcdefg";
	reversePrint(str);
}

int fibonacci(int pos) 
{
	if (pos == 1 || pos == 2)
	{
		return 1;
	}

	return fibonacci(pos - 1) + fibonacci(pos - 2);
}
void test02()
{
	//斐波那契数列
	// 1 1 2 3 5 8 13 21  34  55...
	printf("第9为数字为:%d\n", fibonacci(9)); 
	printf("第10为数字为:%d\n", fibonacci(10));
	printf("第20为数字为:%d\n", fibonacci(20));

}

总结

到这里这篇文章的内容就结束了,谢谢大家的观看,如果有好的建议可以留言喔,谢谢大家啦!

相关推荐
虾球xz15 分钟前
游戏引擎学习第55天
学习·游戏引擎
古希腊掌管学习的神22 分钟前
[LeetCode-Python版]相向双指针——611. 有效三角形的个数
开发语言·python·leetcode
赵钰老师23 分钟前
【R语言遥感技术】“R+遥感”的水环境综合评价方法
开发语言·数据分析·r语言
就爱学编程31 分钟前
重生之我在异世界学编程之C语言小项目:通讯录
c语言·开发语言·数据结构·算法
oneouto32 分钟前
selenium学习笔记(二)
笔记·学习·selenium
sealaugh3236 分钟前
aws(学习笔记第十九课) 使用ECS和Fargate进行容器开发
笔记·学习·aws
Oneforlove_twoforjob1 小时前
【Java基础面试题025】什么是Java的Integer缓存池?
java·开发语言·缓存
emoji1111111 小时前
前端对页面数据进行缓存
开发语言·前端·javascript
炭烤玛卡巴卡1 小时前
学习postman工具使用
学习·测试工具·postman
每天都要学信号1 小时前
Python(第一天)
开发语言·python