1.函数的定义和声明
函数定义的语法:
函数类型 函数名(形式参数表){
函数体
}
函数的声明
函数返回值类型 函数名(变量1数据类型, 变量2数据类型,
变量n数据类型)
函数的调用
函数名(变量1, 变量2,
变量n)
函数调用时注意事项:
- 函数调用的实参个数必须与形参个数相同。
- 实参与形参按照在参数表中的位置一一对应传值,实参与形参的名称是否相同对调用传值无任何影响。
- 实参与形参对应位置上的数据类型应该一致。
- 对于无参数函数,即形参表为void的函数,函数调用时实参表必须为空,不能有任何内容。
当调用函数的位置在被调用函数之后,则可以不进行声明;反之则需要先声明,再调用。
以下案例可以不进行声明:
#include<stdio.h>
double max(double a, double b) {
return a > b ? a:b;
}
void main(){
double a = 1.3;
double b = -5.6;
printf("%.1lf和%.1lf的较大值为%.1lf\n", a, b, max(a, b));
}
以下案例,则需要进行先声明,再调用:
#include<stdio.h>
double max(double, double);
void main(){
double a = 1.3;
double b = -5.6;
printf("%.1lf和%.1lf的较大值为%.1lf\n", a, b, max(a, b));
}
double max(double a, double b) {
return a > b ? a:b;
}
2.递归函数
递归函数:在定义一个函数的过程中直接或间接地调用了被定义的函数本身。
递归函数一般分为两类:公式递归和非公式递归。
//公式递归
#include<stdio.h>
long l_sum(int n) {
//求1至n的和
if (n == 1) return 1;
else{
return l_sum(n - 1) + n;
}
}
//非公式递归
#include<stdio.h>
void hanoi(int n, char a, char b, char c) {
// 汉塔诺问题
if (n == 1) {
printf("%c--->%c\n", a, b);
}else{
hanoi(n - 1, a, c, b);
printf("%c--->%c\n", a, b);
hanoi(n - 1, c, b, a);
}
}
void main(){
hanoi(3, 'A','B' ,'C');
}
3.指针函数
函数返回值是指针类型的函数称为指针函数。语法如下:
数据类型 *函数名称(形参表) {
函数体
}
3.1 指针函数例子1——求字符串数组中最长字符串
#include<stdio.h>
#include<string.h>
char *max_string_len(char *string[], int n){
// 求给定的字符的指针数组中,字符长度最长的字符串。n为指针数组个数
int i, posion, max_l;
posion=0;
max_l=strlen(string[0]);
for (i = 0; i < n; i++){
if (strlen(string[i]) > max_l){
max_l = strlen(string[i]);
posion = i;
}
}
return string[posion];
}
void main(){
char *pstring[4]={"abc", "peculiar", "accent", "rural"};
printf("%s\n", max_string_len(pstring, 4));
}
3.2 指针函数例子1——求学生等级人数
#include<stdio.h>
#define N 6
void main(){
int flag(int, int);
void output(int *, int);
int s1, s2, i;
static int r[5];
for (i = 0; i < N; i++){
scanf("%d%d", &s1, &s2);
r[flag(s1, s2)]++;
}
output(r,5);
}
int flag( int s1, int s2){
// 根据传入的两门课的成绩,求平均值,并返回对应的等级。
int ave;
ave = (s1 + s2) / 2;
if (ave >= 90) return 0;
else if (ave >= 80) return 1;
else if (ave >= 70) return 2;
else if (ave >= 60) return 3;
else return 4;
}
void output(int * p, int n){
// 给定一个指针和整数n,从而实现逐个打印数组中元素的值
int i;
for (i = 0; i < n; i++){
printf("%d ", *(p + i));
}
}
3.3 指针数组和数组指针
- 数组指针:指向数组的指针,指针指向数组首元素地址的指针。例子: int (*p)[5]
- 指针数组: 存放指针的数组,一般用来存放字符串。例子: char *p[5]。
4.函数案例分享
4.1 求1+2+…+n的和
int sum(int n){
//求1+2+...+n的和
int i, s = 0;
for (i = 1; i <= n; i++){
s = s + i;
}
return s;
}
4.2 绘制指定数量的*
void pstar(int n){
// 绘制指定数量的*
for (int i = 1; i <= n; i++){
printf("*");
}
}
4.3 求n的阶乘
long f(int n){
//求n的阶乘
long s = 1l;
for (int i = 1; i <= n; i++){
s = s * i;
}
return s;
}
4.4 绘制n个字符c
void pchar(char c, int n) {
// 绘制n个字符c
for (int i = 1; i <= n; i++){
printf("%c",c);
}
}
4.5 交换两个变量的值
void swap(int *p1, int *p2){ //传的是指针,实际的地址参数,与实参共享空间,实际值会发生改变
int temp;
temp = *p1;
*p1 = *p2;
*p2 = temp;
}
4.6 哥巴赫猜想
任一大于2的偶数都可写成两个质数之和。
#include<stdio.h>
#include<math.h>
int is_prime(int n){ //判断是否为素数,如果是素数返回1,否则0
int i;
for (i = 2; i <= sqrt(n); i++){
if (n % i == 0) return 0;
}
return 1;
}
void main(void){
int i, n;
scanf("%d", &n);
for(i = 3; i < n / 2 ; i += 2){
if (is_prime(i) && is_prime(n - i)){
printf("%d + %d = %d\n", i, n - i, n);
break;
}
}
}