1.函数的概念
函数:founction
c语言的程序代码都是函数组成的
c语言中的函数就是一个完成某项特定的任务的一段代码,这段代码有特殊的写法和调用方法
c语言中我们一般见到两种函数:
.库函数
.自定义函数
2.库函数
有对应的头文件
#include <math.h>
int main()
{double r = sqrt(16);printf("%lf\n", r);return 0;
}用sqrt求平方根一定要添加头文件#include <math.h>
3.自定义函数
ret_type是用来表示函数计算结果的类型,
函数的参数可以是void,明确表示函数没有参数
//函数的定义
int Add(int x, int y)
{int z = 0;z = x + y;return z;// return z把z的值返回到c里面
}int main()
{int a = 0;int b = 0;//输入scanf("%d %d", &a, &b);//计算//int c = a + b;int c = Add(a, b);//使用函数,调用函数,调用函数就跑到1~7行调用函数了,把a传给x,b传给y//输出printf("%d", c);return 0;
}还有种更加简介的写法:
int Add(int x, int y)
{return x + y;// return z把z的值返回到c里面
}
直接返回x+yvoid print(void) //无参数、无返回值 ,前面的是没有返回值,后面的是不需要参数
{printf("hello world");
}
int main()
{print();//直接调用return 0;
}返回类型的地方的void表示函数没有返回值
参数部分的void表示函数不需要参数
4.形参和实参
在函数的使用过程中,把函数分为形参和实参
int Add(int x, int y) // 形参,形式上的存在
{return x+y;
}int main()
{int a = 0;int b = 0;//输入scanf("%d %d", &a, &b);//计算//int c = a + b;int c = Add(a, b);//使用函数,调用函数//输出printf("%d", c);return 0;
}
在调用函数的时候,真实传递给函数额参数叫;实际参数,简称实参,
在这个代码里面,我们把第15行的a和b叫做实参
在函数定义部分,函数名后面的参数叫;形式参数,就是本代码里面的x和y
形参其实是实参的临时拷贝
5.return语句
return后面可以是一个数值,也可以是一个表达式,如果是表达式就先执行表达式,再返回表达式的结果
举个例子:return x + y;//令x=10,y=20,只有算完x+y才会返回值
int test()
{int a = 0;scanf("%d", &a);if (a > 0)return 1;elsereturn -1;
}
int main()
{int ret = test();printf("%d\n", ret);return 0;
}return 语句执行后,函数彻底返回,后面的代码不再执行void test()
{printf("hehe\n");if (1)//条件为真return;//直接返回,printf("haha\n");//被跳过
}
int main()
{test();return 0;
}返回值和返回类型不同会出现数据丢失int test()
{return 3.14;//返回的是浮点数,但被转换成整型了,如果返回值和返回类型不统一则会出现数据丢失
}
int main()
{int r = test();printf("%d\n", r);return 0;
}如果不想出现数据丢失,可以想强制将3.14改变成整型
return (int)3.14;如果函数中存在if等分支的语句,则要保证每种情况下都有return返回,否则会出现编译错误
函数不写返回类型,默认返回的是整型
int test()
test()函数的返回值没写的话,编译器会认为返回的是int类型的值
如果函数要求返回值,但是函数中没有使用return返回值吗,那具体返回什么就不确定了
6.数组做函数参数
void set_arr(int arr[10],int sz)//不需要返回值,所以加一个void,这里之所以用arr[10]是因为,下面把数组传过来了,我们要创建一个数组用来接受,sz也一样,同样需要一个形参去接受
{int i = 0;for (i = 0; i < sz; i++){arr[i] = -1;}
}int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };//写一个函数将一个整型数组的内容,全部置为-1,int sz = sizeof(arr) / sizeof(arr[0]); //算出数组元素的个数set_arr(arr,sz);//传数组 //set_arr设置数组,把arr数组中的sz个数组设置一下,这里不能写arr[10],因为arr[10]是数组中下标为10的那个数,如果要传数组的话,传数组名就行了return 0;
}改变:你想赋什么就赋什么,在46行括号内加上你想要赋的值,然后传到函数里面的set,然后通过34行就可以实现了
void set_arr(int arr[10],int sz,int set)//不需要返回值,所以加一个void,这里之所以用arr[10]是因为,下面把数组传过来了,我们要创建一个数组用来接受,sz也一样,同样需要一个形参去接受
{int i = 0;for (i = 0; i < sz; i++)//通过循环使每一个值都赋上值{arr[i] = set;}
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };//写一个函数将一个整型数组的内容,全部置为-1,int sz = sizeof(arr) / sizeof(arr[0]); //算出数组元素的个数set_arr(arr,sz,-1);//传数组 //set_arr设置数组,把arr数组中的sz个数组设置一下,这里不能写arr[10],因为arr[10]是数组中下标为10的那个数,如果要传数组的话,传数组名就行了return 0;
}void set_arr(int arr[10],int sz,int set)//不需要返回值,所以加一个void,这里之所以用arr[10]是因为,下面把数组传过来了,我们要创建一个数组用来接受,sz也一样,同样需要一个形参去接受
{int i = 0;for (i = 0; i < sz; i++){arr[i] = set;}
}
void print_arr(int arr[10],int sz)
{int i = 0;for (i = 0; i < sz; i++)//sz=10,i<sz就是<=9,也就是标号为9的数字,也就是最后一位数字{printf("%d", arr[i]);}printf("\n");
}int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };//写一个函数将一个整型数组的内容,全部置为-1,int sz = sizeof(arr) / sizeof(arr[0]); //算出数组元素的个数print_arr(arr, sz);//设置之前数组内数字的大小没有变化set_arr(arr,sz,-1);//传数组 //set_arr设置数组,把arr数组中的sz个数组设置一下,这里不能写arr[10],因为arr[10]是数组中下标为10的那个数,如果要传数组的话,传数组名就行了//打印数组的函数print_arr(arr, sz);return 0;
}输出结果为:
12345678910 //改变前
-1-1-1-1-1-1-1-1-1-1 //改变后
数组在传参的时候,实参就写数组名,形参也是数组的形式
实参和形参的名字是可以一样的
函数在设计的时候,一定要功能单一,不要要过于复杂
数组在传参的时候,形参的数组和是实参的数组是同一个数组
形参如果是一维数组,数组大小可以省略不写
形参如果是二维数组,行可以省略,但是列不能省略
//二维数组的案例
void print(int arr[3][5], int r, int c)//这里的3可以省略,但是5不能省略
//这里的r和c是元素个数
{int i = 0;for (i = 0; i < r; i++)//几行//这里的r和c是元素个数,这是个循环{int j = 0;for (j = 0; j < c; j++)//一行有几列{printf("%d ",arr[i][j]);//这里就要用下标访问了,这里的i和j 就是下标,下标的大小要比元素的数小1}printf("\n");}
}int main()
{int arr[3][5] = { 1,2,3,4,5, 2,3,4,5,6, 3,4,5,6,7 };//定义时是3行5列,这里是元素个数//打印二维数组的内容print(arr, 3, 5);return 0;
}
7.嵌套调用和链式访问
//假设我们某年某月有多少天
//只有闰年的2月有29天,需要对y判断是否是闰年//函数只有两种返回
//是闰年
//不是闰年
int is_leap_year(int y)
{if(((y%4==0)&&(y%100!=0)||(y%400==0)))//闰年的判断,return 1;//如果是闰年就返回1elsereturn 0;//不是闰年就返回0
}int get_days_of_month(int y ,int m)//接受年和月//这里我们需要返回一个整型
{int days[] = {0, 31,28,31,30,31,30,31,31,30,31,30,31 };// 0 1 2 3 4 5 6 7 8 9 10 11 下标int day = days[m];if (is_leap_year(y) == 1 && m==2)//如果是闰年就返回一,表示为真的,并且m是二月{day++;//闰年二月的时候,天数+1,为29天}return day;//如果不是二月就返回day}int main()
{int month = 0;int year = 0;scanf("%d %d", &year, & month);//计算某年某月有多少天int day =get_days_of_month(year, month);//这个函数传了年和月printf("%d\n", day);return 0;
}//main函数调用get_days_of_month去计算某年某月有多少天
//get_days_of_month这个函数内部又要通过is_leap_year判断输入的数是否是闰年
//17行的==1可以不写做出改变,用布尔类型来做:
bool is_leap_year(int y) //布尔类型,判断真假
{if (((y % 4 == 0) && (y % 100 != 0) || (y % 400 == 0)))//闰年的判断,return true;//如果是闰年就返回trueelsereturn false;//不是闰年就返回false
}
所谓链式访问就是将一个函数的返回值作为另一个函数的参数,像链条一样将函数串起来就是函数的链式访问int main()
{//printf("%d", printf("%d", printf("%d ", 43)));//输出结果是4321/*int r = printf("hehe");printf("%d\n", r);*///输出结果是hehe4,返回的是4个字符printf("%d", printf("%d ", printf("%d ", 43)));//printf("%d", 43)这个打印出来的是43,打印两个字符返回值是2//printf("%d", printf("%d", 2));//printf("%d", 2) 再屏幕上打印2,打印一个字符返回值是1//printf("%d", 1);打印就是1//所以最终的结果是4321/*先打印43,再打印2,最后打印1*///如果在中间和后面的两个printf里面的%d后面添加一个空格,输出结果就不一样了,变为43 3 2//这种情况的话最先的打印是43加一个空格,返回3个字符//然后第二次打印就是打印3和一个空格,返回2个字符//然后就变成了printf("%d",2),打印2//最终结果就变成43 3 2return 0;
}
8.函数的声明和定义
函数或者变量都要满足,先声明后使用
声明后面要加分号,定义后面不用加分号
//函数的定义是一种特殊的声明
/*int is_leap_year(int y);*///函数的声明,只要函数提前声明了,不管函数定义在前面还是后面,都能使用
int is_leap_year(int );//函数声明的另一种写法,在声明内容中形参的名字是可以不用写的int main(){int year = 0;scanf("%d", year);if (is_leap_year(year)==true)//函数的调用{printf("%d 是闰年", year);}else{printf("%d 不是闰年", year);}return 0;
}//函数的定义int is_leap_year(int y) //布尔类型,判断真假{if (((y % 4 == 0) && (y % 100 != 0) || (y % 400 == 0)))//闰年的判断,return 1;//如果是闰年就返回trueelsereturn 0;//不是闰年就返回false}
新创建一个源文件add.c
新创建一个头文件add.h
源文件add.c放函数的定义实现
头文件add.h放函数的声明
如果要使用的话,就写出头文件打代码
#include "add.h"
写出这行代码,头文件就能正常使用了这样就把函数的声明和定义拆散了,函数定义放在.c文件
函数声明放在头文件我们要把add.h和add.c当做一个模块,只要想使用这个模块,包含头文件即可
把大型复杂的程序,拆分成多个文件的好处
1.团队协作
2.代码模块化,逻辑更加清晰
3.代码的隐藏
局部变量的作用域就是变量所在的局部范围//int main()
//{
// {
// int a = 100;//定义的a只在这个括号内使用,出了这个括号就用不了了
// printf("1 a=%d\n", a);
// }
// //printf("2 a=%d\n", a);
//
//
// return 0;
//}
int main()
{int a = 100;//把a的定义放在外围的括号内处于这个括号的打印都能用{//定义的a只在这个括号内使用,出了这个括号就用不了了 printf("1 a=%d\n", a);}printf("2 a=%d\n", a);return 0;
}
全局变量的作用域是整个工程int a = 100;
int main()
{ {printf("1 a=%d\n", a);}printf("2 a=%d\n", a);return 0;
}
static是静态的意思,可以用来:
1.修饰局部变量
2.修饰全局变量
3.修饰函数extem是用来声明外部符号的ertem:
在ass.c定义int a = 100;这就是定义的全局变量//extern是用来声明外部符号的
笔记.c
extern int a;//变量的声明
int main()
{ {printf("1 a=%d\n", a);}printf("2 a=%d\n", a);return 0;
}
add.c
int a = 100;extern是声明外部符号的,只要这个符号是来自外部的,只要用extern 就可以调用了声明一各变量的方式,知道类型和名字就行了
//void test()//进入函数
//{
// int a = 1;/*将a赋值为1局部变量*/
// a++;//a直接新创建一个值
// printf("%d", a);
//}
//
//int main5()
//{
// for (int i = 0; i < 5; i++)//i从0开始,循环5次
// {
// test();
// }
// return 0;
//}
//为什么这个函数输出是2
//因为每次调用test函数的时候,我们就会定义一次a并赋值为1,然后就是2,
//第二轮循环也一样,在每个循环的开始a都是以1出现的,所以最后打印的结果就是5个2void test()//进入函数
{static int a = 1;//在局部变量前面加static\//输出结果截然不同,结果为:23456a++;printf("%d", a);
}int main()
{for (int i = 0; i < 5; i++)//i从0开始,循环5次{test();}return 0;
}
//为何在局部变量前面添加static后,结果变得不一样了
//static修饰局部变量改变了变量的声明周期
//在test函数中没有static的话,局部变量 是进入这个函数开始,出去这个函数生命周期结束
//没有被static修饰,局部变量的位置是在栈区的,进入这个函数生成,出去这个函数销毁
//被static修饰修饰的变量就不放在栈区了,就放在静态区了,static修饰修饰的变量叫静态变量
//存储在静态区的变量和全局变量一样的,声明周期就和程序的生命周期一样了
//但是本代码中的a的作用域不变,
//ststic修饰的变量,在第二轮循环的时候,使用的变量就是上一次留下来的,而不是新建的变量//如何使用static去修饰变量
//:未来一个变量出了函数,我们还想保留值,等下次进入函数继续使用,就可以使用static去修饰这个变量
//如何使用static去修饰局部变量
//:未来一个变量出了函数,我们还想保留值,等下次进入函数继续使用,就可以使用static去修饰这个变量
// 将全局变量放到其他文件
//函数要先声明再使用,变量也一样
//一个全局变量,要想在其他文件中使用,只需要用extern声明外部符号,就可以使用了
//全局变量是默认带有外部链接属性的,只要用extern引用就行了
extern int g_val;//对变量进行声明
int main()
{printf("%d\n", g_val);return 0;
}
//但是如果想引用其他文件夹的全局变量,但是这个全局变量被static修饰了,
// 那么这个全局变量就不能被引用//static修饰全局变量的时候,改变了全局变量的链接属性
//使得外部链接属性,变成了内部链接属性
//这种变量只能在自己所在的.c文件中使用,其他的.c文件中无法使用//使用建议:如果一个全局变量,只想在所在的源文件内部使用,不想被其他文件发现,
// 就可以使用static修饰
//函数在另一个文件//extern Add(int, int);//声明外部符号就可以使用了
//函数也具有外部链接属性的,只要在其他的.c文件中正确的声明,也可以直接使用int main()
{int a = 0;int b = 0;scanf("%d %d", &a, &b);int c = Add(a, b);printf("c=%d\n", c);return 0;
}//Add(int x, int y)//static修饰函数,让函数外部链接属性变为内部链接属性
// 只能在自己所在的.c文件中使用,其他.c文件中无法使用
//{
// return x + y;
//}