为什么我们会用到函数?
函数的优点:1)提高了代码的复用性。
2)在一个函数中,尽量少定义其他的函数,以便于提高函数的扩展性。
3)写函数讲究得叠代开发。
函数的三要素:函数名,函数的返回值,函数的形参。
函数名:
要做到“自注释性”,即增加代码的可读性, 一般为动宾结构的词构成。在命名的时候,要注意下划线的个数,一般不超过5个
形参:
与实参的区别:
1)实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值。(传出参数例外,没有初始化,从函数里传出,相当于函数的返回值,与之相对应的是传入参数,传入参数是初始化了一个值,放到了函数里。)
如上图所示:在主函数中虽没有定义num为多少,但是在func函数中把num定义为5,这样子打印出来为num = 5,相当于返回num的值。
2)实参和形参在数量上,类型上、顺序上应严格一致,否则就会发生类型不匹配的错误。
在一般传值调用的机制中只能把实参传送给形参,而不能把形参的值反向地传送给实参。因此在函数调用过程中,形参值发生改变,而实参中的值不会变化。而在引用调用的机制当中是将实参引用的地址传递给了形参,所以任何发生在形参上的改变实际上也发生在实参变量上。
3)形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量。
以下的程序是交换a,b两个参数的值
函数名是指针常量,保存的是函数存放的地址(入口地址)
详细过程如下:
exchange(&a,&b);
调用函数的详细步骤:1)通过函数名找到函数的入口地址
2)给形参分配内存空间
3)传值,把实参变量对应的空间的值传给形参
4)执行函数体里的语句
5)函数返回并释放
对于传参的总结:读(不修改实参的值)实参变量对应的内存空间值,传实参变量名。
写(改变实参的值)实参变量对应的内存空间值,传实参变量地址。
打印出来的结果为“ello world”,因为传的是地址,所以改变了ptr的值。
结果为段错误,没把实参的地址传过去,形参分配的空间没有传过来,ptr为野指针。
改成这样就输出为“hello world”了,要把实参的地址传过去。
主函数的形参:
int main(int argc,int *argv[]);用于命令行,argc表示几个参数(包含命令),参数存储在argv[]里
返回值
输出为”hello world”,char * ptr = “hello world”存储在数据区,它的生命周期为整个过程,到程序运行完才会释放,若变为 char src[3][100] = “hello world”,存储在栈空间,在函数执行完后,就会释放,所以一般会输出不完全ptr。在函数的返回值中,不能返回局部变量的地址。
exit(1)表示退出整个程序,用来处理异常值,头文件为<stdlib.h>。
前面加printf说明一下是因为什么而退出的,也可以用echo $?来打印内容。
exit()就是退出,传入的参数是程序退出时的状态码,0表示正常退出,其他表示非正常退出,一般都用-1或者1,标准C里有EXIT_SUCCESS和EXIT_FAILURE两个宏,用exit(EXIT_SUCCESS);可读性比较好一点。
括号里的数据可以随便赋值,可以标志哪里出错
Return 0表示系统正常的返回值,告诉系统程序执行完了。其他非0值表示不正常退出,系统会争对这个做相应的处理。可以提高系统的运行效率。