函数的定义:子程序 是一个大型程序中的某部分代码,由一个或多个语句块组成,它负责完成某项特定任务,相较于其他代码,具备相对的独立性
2.库函数 eg:打印函数:printf 字符串拷贝:strcpy 计算n的k次方:pow函数
3.自定义函数
4.函数参数
5.函数调用
6.函数的嵌套调用和链式访问
7.函数的声明和定义
8.☆☆☆☆☆☆ 函数递归 ☆☆☆☆☆☆
函数具备相对独立性 由一个或者多个语句块组成,负责完成某项特定任务
一般会有输入参数并由返回值,提供对 过程 的封装和细节的隐藏,这些代码通常被集成为软件库
函数分为两类:1.库函数 2.自定义函数
库函数:C语言提供的标准代码,为了提高效率和可移植性, 编译器厂商 提供了一系列 库函数
C语言标准中约定好,由编译器的厂商提供实现
eg.C语言标准规定:strlen 函数的功能:求字符串的长度 函数名:strlen 参数:const char* str 返回类型 size_t
库函数:标准库中规定好的函数
使用库函数:提高效率,帮助程序员软件开发
头文件中包含函数
常用的库函数有:
1.IO函数(输入输出函数)
2.字符串操作函数
3.内存操作函数
4.时间/日期函数
5.数学函数(math.h)
6.其他库函数
次方,pow 次方函数:
#include <math.h> 对应的数学的头文件
int main() { long long int n = (long long int)pow(2,6);//二的五次方 double a = (double)pow(5.2, 3.0); printf("%lf\n", a); printf("%lld\n", n); return 0; }
#include <string.h>
strcpy函数: 复制字符串 把一个数组指向的字符串拷贝在另一个数组中
int main() { char arr1[20] = { 0 }; char arr2[] = "hello C"; strcpy(arr1, arr2);//将数组2 arr2中的字符拷贝到数组1 arr1中打印 arr1目标空间的地址 printf("%s\n", arr1); printf("%s\n", strcpy(arr1, arr2)); //返回目标空间的地址 return 0; }
memest函数 内存设置函数 内存:memory
int main() { char arr[] = "hello C";//memset函数将hello C 中 hello全部改变 memset(arr, 'c', 7);//arr中的数组 改为c 共五个字节 char arr1[] = "一切都会好的"; memset(arr1, 'l', 4); printf("%s", arr1); printf("%s", arr); return 0; }
\0是字符串结束标志
库函数查询工具的使用:www.cplusplus.com
/*************自定义函数**************/
函数的组成:函数名 函数参数 返回类型 函数体
eg.写一个函数找出两个整数中的最大值 返回较大值整数
int main() { printf("请您输入两个整数:"); int a, b; scanf("%d %d", &a,&b); int c = max(a, b); printf("%d\n", c); printf("%d\n", max(a, b)); return 0; }//库函数 开箱即用
编写一个函数 比较两个数中的最大值
int Max(int a, int b) { if (a >= b) { printf("%d", a); } else { printf("%d", b); } } int main() { int a = 0; int b = 0; scanf("%d %d", &a, &b); Max(a, b); return 0; }
错误传参方法:
写一个函数交换两个整形变量内容
形式参数 形参 赋值临时变量第三个瓶子
void change(int x, int y) { int t = 0; t = x; x = y; y = t; } int main() { int a, b; scanf("%d %d", &a, &b); printf("交换前:%d %d", a, b); //实际参数 实参 change(a, b); //函数调用的时候就,将实参传递给形参 //形参是实参的一份临时拷贝 //对形参的修改不会改变实参的大小 printf("交换后:%d %d", a, b); return 0; }
int main() { int num = 10; int* p = #//将num的地址传给指针变量,让指针变量进行保存,函数内外部之间没有联系 //通过指针变量赋值,将地址联系起来,从而改变原参数的值。 *p = 20; printf("%d\n", num); return 0; }
正确传参方式:
写一个函数交换两个整形变量内容
*变量存地址指针 地址变量
void change(int*a, int*b)//交换p1 p2顺序 交换地址,传递地址 交换顺序 *a *b代表指针代表地址 { int tmp = 0;//创建临时变量 tmp = *a;//tmp=num1; *a = *b; //num1=num2; *b = tmp;//num2=tmp; } int main() { int a = 0; int b = 0; scanf("%d %d", &a, &b); printf("交换前:num1 = %d num2 = %d", a, b); //实际参数 实参 change(&a, &b); //函数调用的时候就,将实参传递给形参 //形参是实参的一份临时拷贝 //对形参的修改不会,改变实参的大小 printf("交换后:num1 = %d num2 = %d", a, b); return 0; }
实参:真实传给函数的参数,叫做实参,可以是常量,变量,表达式,函数等
形参:指函数名后括号中的变量。因为形参只有在函数被调用的过程中才实例化(分配内存单元)
不调用函数时,形参只是一个形式,函数只有被调用的时候,形参才会被分配变量,形参相当于局部变量,只在函数内部有效
函数不被调用时,函数中的形参变量是一个模板,函数只有在被调用时,才会给参数分配空间
形参实例化后相当于实参的一份临时拷贝
例如、
int Max(int x, int y) { if (x > y) { return x; } else { return y; } } void print() { printf("一切都会好的\n"); } int main() { int num1, num2; scanf("%d %d", &num1, &num2); int b = Max(1 * 9, 81 / 3); printf("b:%d\n", b); int a = Max(num1, num2); printf("max:%d\n", a); print(); return 0; }
函数的调用:
传值调用 传址调用:
传值调用:只需要求值,和值有关,形参只能临时拷贝,对形参的修改不会影响实参
传址调用:把函数外部创建变量的内存地址传递给函数参数的一种调用方式,也就是函数内部可以直接操作函数外部的变量,
当你需要在函数内部修改来自函数外部的变量时,用传址调用,但是传址的本质也是传值调用
区分只有场景的不同
写一个函数判断是不是素数 是素数返回1,不是素数返回.
int is_prime(int n) { //拿2~sqrt(n)开平方n之间数字试除 sqrt:开平方 int i = 0;//判断变量 for(i=2;i<=sqrt(n);i++) { if (n % i == 0) return 0;//直接返回,后面的循环不再执行 //return比break效果更强 } return 1; } int main() { int i = 0; int count = 0; for (i = 100; i <= 200; i++) { if (is_prime(i)) { count++; printf("%d ", i); } } printf("\ncount=%d\n", count); return 0; }
Bool类型进行判断 <stdbool.h>
_Bool 类型的变量只有两种取值 true和false
bool Judge(int a) { int i = 0; for (i = 2; i <= sqrt(a); i++) { if (a % i == 0) { return false; } } return true; } int main() { int count = 0; int a = 0; for (a = 100; a <= 200; a++) { if (Judge(a)) { count++; printf("%d ", a); } } printf("一共有:%d 个素数\n", count); return 0; }
写一个函数判断是不是闰年
打印1000—2000年之间的闰年
方法1:
bool year(int y) { if ((y % 400 == 0 ) || (( y % 4 == 0) && (y % 100 != 0))) return true; else return false; } int main() { int y = 0; int count = 0; for (y = 1000; y <= 2000; y++) { if (Year(y)) { count++; printf("%d ", y); } } printf("count=%d\n", count); return 0; }
方法 2:
int Year(int y) { if ((y % 400 == 0) || ((y % 4 == 0) && (y % 100 != 0))) return 1; else return 0; } int main() { int y = 0; int count = 0; for (y = 1000; y <= 2000; y++) { if (Year(y)) { count++; printf("%d ", y); } } printf("count=%d\n", count); return 0; }
键盘输入一个数,判断是闰年平年 方法1:
int Year(int y) { if ((y % 400 == 0) || ((y % 4 == 0) && (y % 100 != 0))) return 1; else return 0; } int main() { int year = 0; scanf("%d", &year); if (Year(year)) { printf("%d是闰年\n", year); } else { printf("%d不是闰年\n", year); } return 0; }
键盘输入一个数,判断是闰年平年 方法2:
bool Judge(int a) { if ((a % 400 == 0) || ((a % 4 == 0) && (a % 100 != 0))) return true; else return false; } int main() { int year = 0; printf("请您输入一个年份:\n"); scanf("%d", &year); if (Judge(year)) { printf("%d是闰年\n", year); } else { printf("%d不是闰年\n", year); } return 0; }
写一个函数,实现一个整形有序数组的二分查找
找到了就返回下标,找不到返回0;
结果容易溢出,此方法不好。
有序数组:用二分查找
int beat(int arr[], int k, int sz) { int left = 0; int right = sz - 1;//元素数-1 while (left<=right) { //int mid = (left + right) / 2; int mid = left + (right - left) / 2;//代码更加抗揍 if (arr[mid] < k) { left = mid + 1; } else if (arr[mid] > k) { right = mid - 1; } else { return mid; } } return -1; } int main() { int arr[] = { 1,2,3,4,5,6,7,8,9,10 }; int k = 0; scanf("%d", &k);//输入要查找的值 //FindNum(); 函数名 int sz = sizeof(arr) / sizeof(arr[0]);//数组个数 sizeof(arr[]) int ret = beat(arr, k, sz);//k:在数组中寻找的元素 if (ret == -1) { printf("找不到\n"); } else { printf("找到了,下标是%d\n", ret); } return 0; }
数组元素过多 防止数据溢出 改造方法:
改造方法
值太长,结果溢出。
int main()
{
int num1 = 2147483646;
int num2 = 2147483644;
int vag = num1 + (num1 - num2) / 2;
printf("%d\n", vag);
return 0;
}
写一个函数,每调用一次这个函数,就会将num的值加一,方法1:
void Add(int* p) { *p = *p + 1; } int main() { int num = 0; Add(&num); printf("%d\n", num); Add(&num); printf("%d\n", num); Add(&num); printf("%d\n", num); return 0; }
方法2:
int Add(int n) { return n + 1; } int main() { int n = 0; int num = 0; num = Add(num); printf("%d\n", num); num = Add(num); printf("%d\n", num); for(n=1;n<=15;n++) { num = Add(num); printf("%d\n", num); } return 0; }
函数的嵌套和调用
函数可以嵌套调用,但是函数不能嵌套定义,不能在一个函数的定义在再定义一个函数
一个函数中不能定义另一个函数,函数不能嵌套定义
函数之间可以嵌套调用 但是函数之间不能嵌套定义
链式访问:把一个函数的返回值当作另一个函数的参数 叫做链式访问
int main() { int len = strlen("abc"); printf("%d\n", len); printf("%d\n", strlen("abc")); char arr1[20] = { 0 };//abc\0...... char arr2[] = "abc"; printf("%d\n", strlen(strcpy(arr1, arr2))); //strcpy函数将arr2的地址赋值给arr1.strlen函数计算赋值后字符串的长度 return 0; }
eg:链式访问
int main() { printf("%d", printf("%d", printf("43"))); //打印43时返回2,打印2时返回1,利用了函数的链式访问,把一个函数的返回值当作另一个函数的参数 叫做链式访问 //4321 返回值4321 利用了链式访问 不能加入\n printf("\n"); printf("%d ", printf("%d ", printf("43 "))); //打印4332 return 0; }