【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));

}

总结

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

相关推荐
今天我又学废了14 分钟前
Scala学习记录,List
学习
杨荧19 分钟前
【JAVA毕业设计】基于Vue和SpringBoot的服装商城系统学科竞赛管理系统
java·开发语言·vue.js·spring boot·spring cloud·java-ee·kafka
白子寰25 分钟前
【C++打怪之路Lv14】- “多态“篇
开发语言·c++
王俊山IT37 分钟前
C++学习笔记----10、模块、头文件及各种主题(一)---- 模块(5)
开发语言·c++·笔记·学习
为将者,自当识天晓地。39 分钟前
c++多线程
java·开发语言
小政爱学习!41 分钟前
封装axios、环境变量、api解耦、解决跨域、全局组件注入
开发语言·前端·javascript
k09331 小时前
sourceTree回滚版本到某次提交
开发语言·前端·javascript
神奇夜光杯1 小时前
Python酷库之旅-第三方库Pandas(202)
开发语言·人工智能·python·excel·pandas·标准库及第三方库·学习与成长
Themberfue1 小时前
Java多线程详解⑤(全程干货!!!)线程安全问题 || 锁 || synchronized
java·开发语言·线程·多线程·synchronized·
plmm烟酒僧1 小时前
Windows下QT调用MinGW编译的OpenCV
开发语言·windows·qt·opencv