WHCSRL 技术网

[C语言] 函数:库函数--自定义函数--函数的调用--练习--详解<个人>------(1)

文章目录

  • 前言
  • 一. 函数是什么?
  • 二. C语言中函数的分类
  • 三. 库函数
  • 四. 自定义函数
  • 五. 函数的调用
  • 六. 练习
  • 总结



前言

本节内容主要记录函数的相关内容,其中包括:库函数,自定义函数,函数的调用,函数的习题练习。

在上节的循环语句中加入了 goto语句,有兴趣的朋友们可以去了解了解,有一个小整蛊哦。

循环语句链接:

[ C语言 ]循环语句---while---for---do while---详解及练习题巩固_小白又菜的博客-CSDN博客

本篇文章开始之前,首先感谢大家的关注和支持,让我人生首次在一件事情上粉丝破千!!!(真的想象不到)

         其实我开始写博客的初衷是记录自己的学习过程,通过写博客的方式对知识进行再次总结,加深自己对知识的记忆,我现在的目的如此,以后仍然是!每每看到关注量,浏览量这些数据增加,都会让我变得开心,这是大家给予我的一份肯定。在此,感谢各位一直以来的支持,我以后也会对自己的知识记录更加详细,更加丰富,在记录我进步的同时,希望大家通过看到我的博客也会进步。我们一起朝着自己的大厂梦前进吧!!!

 话不多说,上车!!!

———————————————————————————————————————————                                                            ****正文开始****

一. 函数是什么?

维基百科中对函数的定义:子程序
在计算机科学中,子程序(英语:Subroutine, procedure, function, routine, method,
subprogram, callable unit),是一个大型程序中的某部分代码, 由一个或多个语句块组
成。它负责完成某项特定任务,而且相较于其他代 码,具备相对的独立性。
一般会有输入参数并有返回值,提供对过程的封装和细节的隐藏。这些代码通常被集成为软
件库。

二. C言中函数的分类。

1.   库函数
2.   自定义函数

三. 库函数

Q : 为什么会有库函数?

A :

1. 我们知道在我们学习C语言编程的时候,总是在一个代码编写完成之后迫不及待的想知道结果,想把这个结果打印到我们的屏幕上看看。这个时候我们会频繁的使用一个功能:将信息按照一定的格式打印到屏幕上(printf)。

2. 在编程的过程中我们会频繁的做一些字符串的拷贝工作(strcpy)。

3. 在编程是我们也计算,总是会计算nk次方这样的运算(pow)。

像上面我们描述的基础功能,它们不是业务性的代码。我们在开发的过程中每个程序员都可能用的到,为了支持可移植性和提高程序的效率,所以C 语言的基础库中提供了一系列类似的库函数,方便程序员进行软件开发。
那怎么学习库函数呢?
会查询工具: MSDN(Microsoft Developer Network)
在这里给大家分享两个链接:
2 >  cppreference.com       

i>举例:strlen

代码:

  1. #include <stdio.h>
  2. #include <string.h>
  3. int main() {
  4. char arr[] = "abc";
  5. // size_t -> unsigned int
  6. size_t len = strlen(arr);
  7. printf("%%u ", len);
  8. // %%d - 有符号
  9. // %%u - 无符号 (没有负数)
  10. return 0;
  11. }

ii>举例:strcpy

代码演示:

  1. #include <stdio.h>
  2. #include <string.h>
  3. int main() {
  4. char arr1[20] = { 0 };// 目的地
  5. char arr2[] = "hello" ;// 源数据
  6. strcpy(arr1, arr2);
  7. printf("%%s ",arr1);
  8. return 0;
  9. }

结果:

我们发现本来是空的arr1中拷贝了arr2中的"hello"

iii>举例:memset

代码演示:

  1. #include <stdio.h>
  2. #include <string.h>
  3. int main(){
  4. char arr[] = "hello world";
  5. memset(arr, 'x', 5);
  6. printf("%%s ", arr);
  7. return 0;
  8. }

演示结果:

简单的总结,C语言常用的库函数都有:
IO函数
字符串操作函数
字符操作函数
内存操作函数
时间/日期函数
数学函数
其他库函数

四. 自定义函数

1> 自定义函数存在的意义:

由于库函数不能干所有的事情,所以更加重要的是自定义函数

2>函数的组成

自定义函数和库函数一样,有函数名,返回值类型和函数参数。
但是不一样的是这些都是我们自己来设计。

  1. //***自定义函数***
  2. // 函数返回类型 函数名 参数
  3. // ret_type fun_name(para1,*)
  4. // { //
  5. // statement; // 语句项 //函数体
  6. // } //
  7. ret_type 返回类型
  8. fun_name 函数名
  9. para1    函数参数

3>举例

一:写一个函数可以找出两个整数中的最大值。

代码:

  1. // 写一个函数,找出两个数的最大值
  2. //返回类型 函数名 参数
  3. int get_max(int x, int y) {
  4. int z = 0;
  5. z = (x > y ? x : y);
  6. return z;
  7. }
  8. int main() {
  9. int a = 0;
  10. int b = 0;
  11. scanf("%%d %%d", &a, &b);
  12. int m = get_max(a, b);
  13. printf("%%d ", m);
  14. return 0;
  15. }

演示结果:

二:写一个函数可以交换两个整形变量的内容。

按照我们正常的书写代码如下:

  1. void Swap1(int x, int y) {
  2. int temp = 0;
  3. temp = x;
  4. x = y;
  5. y = temp;
  6. }
  7. int main() {
  8. int a = 0, b = 0;
  9. scanf("%%d %%d", &a, &b);
  10. // 交换两个变量
  11. printf("交换前a = %%d, b = %%d ", a, b);
  12. Swap1(a,b); // 传值调用
  13. printf("交换后a = %%d, b = %%d ", a, b);
  14. return 0;
  15. }

正确的写法:

  1. 正确写法 使用指针
  2. void Swap2(int* pa,int* pb) {
  3. int t = 0;
  4. t = *pa;
  5. *pa = *pb;
  6. *pb = t;
  7. }
  8. int main() {
  9. int a = 0, b = 0;
  10. scanf("%%d %%d", &a, &b);
  11. // 交换两个变量
  12. printf("交换前a = %%d, b = %%d ", a, b);
  13. Swap2(&a,&b); // 传址调用
  14. printf("交换后a = %%d, b = %%d ", a, b);
  15. return 0;
  16. }

结果演示:

我们观察这段代码发现,a和b并没有交换。

失败的原因是:a 和 b 是实参 , x 和 y 是形参。当函数调用的时候,实参传给形参,形参其实是实参的一份临时拷贝。所以对形参的修改,不会影响实参。

实际参数(实参):
真实传给函数的参数,叫实参。
实参可以是:常量、变量、表达式、函数等。
无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形
参。
形式参数(形参):
形式参数是指函数名后括号中的变量,因为形式参数只有在函数被调用的过程中才实例化(分配内存单元)
所以叫形式参数。形式参数当函数调用完成之后就自动销毁了。因此形式参数只在函数中有效
上面 Swap1 Swap2 函数中的参数 x y px py 都是 形式参数 。在 main 函数中传给 Swap1 的 a ,b和传给 Swap2 函数的 &a  &b  实际参数
这里我们对函数的实参和形参进行分析:

代码对应的内存分配如下:

这里可以看到 Swap1 函数在调用的时候, x y 拥有自己的空间,同时拥有了和实参一模一样的内容。
所以我们可以简单的认为: 形参实例化之后其实相当于实参的一份临时拷贝

五. 函数的调用

1>传值调用

函数的形参和实参分别占有不同内存块,对形参的修改不会影响实参。

2>传址调用

传址调用是把函数外部创建变量的内存地址传递给函数参数的一种调用函数的方式。
这种传参方式可以让函数和函数外边的变量建立起真正的联系,也就是函数内部可以直接操作函数外部的变量。

六. 练习

一:写一个函数可以判断一个数字 是不是素数

  1. //is_prime()
  2. //是素数返回 1 ,不是素数返回 0
  3. int is_prime(int n) {
  4. // 2 ~ n-1 的数字试除
  5. int j = 0;
  6. for (j = 2; j <= sqrt(n); j++) {
  7. if (n %% j == 0) {
  8. return 0;
  9. }
  10. }
  11. return 1;
  12. }
  13. int main() {
  14. int i = 0;
  15. for (i = 100; i <= 200; i++) {
  16. // 判断 i 是否为素数 - 如果是素数就打印
  17. // i 取 100 - 200
  18. if (is_prime(i) == 1) {
  19. printf("%%d ", i);
  20. }
  21. }
  22. return 0;
  23. }

演示结果:

二:写一个函数判断一年是不是闰年

  1. //写一个函数判断一年是不是闰年
  2. //是闰年 返回1 不是闰年 返回0
  3. int is_leap_year(int y) {
  4. //if (y %% 100 != 0 && y %% 4 == 0 || y %% 400 == 0) {
  5. // return 1;
  6. //}
  7. //else {
  8. // return 0;
  9. //}
  10. return (y %% 100 != 0 && y %% 4 == 0 || y %% 400 == 0);
  11. }
  12. int main() {
  13. int y = 0;
  14. for (y = 1000; y <= 2000; y++) {
  15. // 判断 y 是不是闰年
  16. if (is_leap_year(y) == 1) {
  17. printf("%%d ",y);
  18. }
  19. }
  20. return 0;
  21. }

代码演示结果:

三: 写一个函数 ,实现一个整形有序数组的二分查找

  1. //写一个函数 ,实现一个整形有序数组的二分查找
  2. //如果找到了返回 下标
  3. //如果找不到返回 -1
  4. int binary_search(int arr[], int k,int sz) {
  5. int left = 0;
  6. int right = sz - 1;
  7. while (left<=right) {
  8. int mid = (left + right) / 2;
  9. if (arr[mid] < k) {
  10. left = mid + 1;
  11. }
  12. else if (arr[mid] > k)
  13. {
  14. right = mid - 1;
  15. }
  16. else if (left == right) {
  17. return mid;
  18. }
  19. }
  20. return -1;
  21. }
  22. int main() {
  23. int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
  24. int k = 7;
  25. int sz = sizeof(arr) / sizeof arr[0];
  26. // 40 / 4
  27. // 数组arr传给binary_search函数的时候,其实传递的arr数组首元素的地址
  28. int ret = binary_search(arr,k,sz); // 在arr数组中找k
  29. if (-1 == ret) {
  30. printf("找不到 ");
  31. }
  32. else {
  33. printf("找到了,下标是%%d ", ret);
  34. }
  35. return 0;
  36. }

代码演示结果:

四:实现一个函数,打印乘法口诀表,口诀表的行数和列数自己指定

如:输入9,输出9 * 9口诀表,输出12,输出12 * 12的乘法口诀表。

  1. void Multiplication_formula_table(int i, int j,int n) {
  2. for (i = 1; i <= n; i++) {
  3. for (j = 1; j <= i; j++) {
  4. printf("%%d*%%d=%%d ", j, i, i* j);
  5. }
  6. printf(" ");
  7. }
  8. }
  9. int main() {
  10. int n = 0;
  11. scanf("%%d",&n);
  12. int i = 0;
  13. int j = 0;
  14. Multiplication_formula_table(i ,j,n);
  15. return 0;
  16. }

结果演示:

                 

                                                     ****正文结束**** 

——————————————————————————————————————————


总结

本篇文章的内容就记录完毕,由于本人技术水平有限,若各位读者发现错误,欢迎及时纠正。

推荐阅读