【C语言-函数1】

简介: 【C语言-函数1】



什么是函数

数学中我们常见到函数的概念。但是你了解C语言中的函数吗?

子程序

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

C语言中函数的分类

1. 库函数

为什么会有库函数?

  1. 我们知道在我们学习C语言编程的时候,总是在一个代码编写完成之后迫不及待的想知道结果,想把这个结果打印到我们的屏幕上看看。这个时候我们会频繁的使用一个功能:将信息按照一定的格 式打印到屏幕上(printf)。
  2. 在编程的过程中我们会频繁的做一些字符串的拷贝工作(strcpy)。
  3. 在编程是我们也计算,总是会计算n的k次方这样的运算(pow)。

像上面我们描述的基础功能,它们不是业务性的代码。我们在开发的过程中每个程序员都可能用的到, 为了支持可移植性和提高程序的效率,所以C语言的基础库中提供了一系列类似的库函数,方便程序员 进行软件开发。

那怎么学习库函数呢? 这里我们简单的看看:http://www.cplusplus.com

库函数是C语言标准中约定好的,由编译器的厂商提供实现

C语言标准规定:

函数的功能- 字符串的长度

函数名- strlen

参数- const char*str

返回类型- size_t

简单的总结,

C语言常用的库函数都有:

IO函数

C语言提供了一组标准的输入输出(I/O)函数,用于读取输入和输出结果。这些函数包括:

  1. printf():用于格式化输出到标准输出设备(通常是终端)。
  2. scanf():用于从标准输入设备(通常是键盘)读取输入并按指定格式进行解析。
  3. getchar():用于从标准输入设备读取单个字符。
  4. putchar():用于将单个字符输出到标准输出设备。
  5. gets():用于从标准输入设备读取一行字符串(不安全,不推荐使用)。
  6. puts():用于将字符串输出到标准输出设备并自动添加换行符。
  7. fgets():用于从文件或标准输入设备读取一行字符串(更安全的替代gets())。
  8. fputs():用于将字符串输出到文件或标准输出设备。

这些函数都属于stdio.h头文件中定义的标准I/O函数。除了上述函数,C语言还提供了其他一些用于文件操作的函数,如fopen()、fclose()、fread()、fwrite()等。

字符串操作函数

在语言中,字符串是以字符数组的形式的,C语言提供了一常用的字符串操作函数,用于对字符串进行处理和操作。以下是一些常见的C语言字符串操作函数:

1. strlen(s):返回字符串s的长度(不包括空字符'\0')。

2. strcpy(dest, src):将字符串src复制到dest中,并返回dest。

3. strncpy(dest, src, n):将字符串src的前n个字符复制到dest中,并返回dest。

4. strcat(dest, src):将字符串src连接到dest的末尾,并返回dest。

5. strncat(dest, src, n):将字符串src的前n个字符连接到dest的末尾,并返回dest。

6. strcmp(s1, s2):比较字符串s1和s2的大小,返回整数值表示比较结果。

7. strncmp(s1, s2, n):比较字符串s1和s2的前n个字符的大小,返回整数值表示比较结果。

8. strchr(s, c):在字符串s中搜索字符c的第一次出现位置,并返回指向该位置的指针。

9. strrchr(s, c):在字符串s中搜索字符c的最后一次出现位置,并返回指向该位置的指针。

10. strstr(s1, s2):在字符串s1中搜索子串s2的第一次出现位置,并返回指向该位置的指针。

11. strtok(str, delimiters):将字符串str按照分隔符delimiters进行分割,并返回分割后的子串。

这些字符串操作函数可以帮助我们处理和操作字符串,如获取字符串长度、复制字符串、连接字符串、比较字符串、查找字符或子串等。需要注意的是,在使用这些函数之前,需要包含相应的头文件(如string.h)。下面是一个简单的示例程序,演示了如何使用一些字符串操作函数:

 

#include <stdio.h>
#include <string.h>
int main() {
    char str1[20] = "Hello";
    char str2[20] = "World";
    printf("Length of str1: %d\n", strlen(str1));
    strcpy(str1, str2);
    printf("After strcpy, str1: %s\n", str1);
    strcat(str1, " ");
    strcat(str1, "C Language");
    printf("After strcat, str1: %s\n", str1);
    if (strcmp(str1, str2) == 0) {
        printf("str1 and str2 are equal\n");
    } else {
        printf("str1 and str2 are not equal\n");
    }
    char *ptr = strchr(str1, 'C');
    if (ptr != NULL) {
        printf("Found 'C' at position: %ld\n", ptr - str1);
    }
    char *token = strtok(str1, " ");
    while (token != NULL) {
        printf("Token: %s\n", token);
        token = strtok(NULL, " ");
    }
    return 0;
}

此程序展示了strlen()、strcpy()、strcat()、strcmp()、strchr()和strtok()函数的使用,可以运行并观察输出结果。

字符操作函数
  1. isalpha(c):判断字符c是否为字母。
  2. isdigit(c):判断字符c是否为数字。
  3. isalnum(c):判断字符c是否为字母或数字。
  4. islower(c):判断字符c是否为小写字母。
  5. isupper(c):判断字符c是否为大写字母。

这些函数都是以单个字符作为参数,并返回一个非零值(真)或零值(假)来表示判断结果。它们通常用于字符的验证和转换操作,可以帮助程序员处理和操作字符数据。在使用这些函数之前,需要包含相应的头文件(如ctype.h)

内存操作函数  

C语言提供了一组内存操作函数,用于对内存进行操作和管理。以下是一些常用的C语言内存操作函数:

  1. memcpy(dest, src, n):将源内存区域src的前n个字节复制到目标内存区域dest中。
  2. memmove(dest, src, n):将源内存区域src的前n个字节复制到目标内存区域dest中,可以处理内存重叠的情况。
  3. memset(ptr, value, n):将指针ptr指向的内存区域的前n个字节设置为指定的值value。
  4. memcmp(ptr1, ptr2, n):比较两个内存区域ptr1和ptr2的前n个字节,返回一个整数值表示比较结果。
  5. memchr(ptr, value, n):在指针ptr指向的内存区域的前n个字节中搜索指定的值value,并返回指向该值的指针。
  6. calloc(num, size):为num个大小为size的连续内存块分配空间,并将其初始化为0,返回指向分配内存的指针。
  7. malloc(size):分配大小为size的连续内存块,并返回指向分配内存的指针。
  8. realloc(ptr, size):重新分配指针ptr指向的内存块的大小为size,并返回指向重新分配内存的指针。
  9. free(ptr):释放之前通过malloc()、calloc()或realloc()函数分配的内存块。

这些函数可以用于动态内存管理、内存复制、内存比较和内存设置等操作。需要注意的是,在使用这些函数时,需要包含相应的头文件(如string.h、stdlib.h)。

下面是一个简单的示例程序,演示了如何使用一些内存操作函数:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main() {
    char str1[] = "Hello";
    char str2[10];
    memcpy(str2, str1, strlen(str1)+1);
    printf("Copied string: %s\n", str2);
    memset(str2, '*', strlen(str2));
    printf("After memset: %s\n", str2);
    int arr1[] = {1, 2, 3, 4, 5};
    int arr2[5];
    memcpy(arr2, arr1, sizeof(arr1));
    printf("Copied array: ");
    for (int i = 0; i < sizeof(arr2)/sizeof(arr2[0]); i++) {
        printf("%d ", arr2[i]);
    }
    printf("\n");
    int* ptr = (int*)malloc(5 * sizeof(int));
    if (ptr == NULL) {
        printf("Memory allocation failed\n");
        return 1;
    }
    for (int i = 0; i < 5; i++) {
        ptr[i] = i + 1;
    }
    printf("Dynamically allocated array: ");
    for (int i = 0; i < 5; i++) {
        printf("%d ", ptr[i]);
    }
    printf("\n");
    free(ptr);
    return 0;
}

此程序展示了memcpy()、memset()、malloc()和free()函数的使用,可以运行并观察输出结果。

时间/日期函数

例如time()、strftime()用于时间和日期的获取和格式化。

time.h是C/C++中的日期和时间头文件。用于需要时间方面的函数。下面分享time.h头文件中几个常用函数的用法:

数学函数  

例如sqrt()、sin()用于数学运算和三角函数计算。

abs()函数 函数原型: int abs(int i)

功能: 求整数的绝对值

labs()函数 函数原型:long labs(long i)

功能:求长型整数的绝对值

fabs()函数 函数原型:double fabs(float i)

功能:求浮点数的绝对值

floor()函数 函数原型:double floor(double x)

功能:求不大于x的最大整数

ceil()函数 函数原型:double ceil(double x)

功能:求不小于x的最小整数

sqrt()函数 函数原型:double sqrt(double x)

功能:求x的平方根。

log()函数 函数原型:double log(double x)

功能:求x的自然对数

log10()函数 函数原型:double log10(double x)

功能:求x的常用对数

pow()函数 函数原型:double pow(double x,double y)

功能:求x的y次方

pow10()函数 函数原型:double pow(double x)

功能:求10的x次方

exp()函数 函数原型:double exp(double x)

功能:求e的x次方

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

版权声明:本文为CSDN博主「辉小歌」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/qq_46527915/article/details/105011484

其他库函数

例如fopen()、fwrite()用于文件的打开和写入操作。

注:

但是库函数必须知道的一个秘密就是:使用库函数,必须包含#include对应的头文件。

这里对照文档来学习上面几个库函数,目的是掌握库函数的使用方法。

如何学会使用库函数?

需要全部记住吗?No

需要学会查询工具的使用:

MSDN(Microsoft Developer Network)

www.cplusplus.com http://en.cppreference.com(英文版)

http://zh.cppreference.com(中文版)

自定义函数

自定义函数和库函数一样,有函数名,返回值类型和函数参数。

但是不一样的是这些都是我们自己来设计。

这给程序员一个很大的发挥空间。

函数的组成:

函数名 参数 返回类型 函数体

我们举一个例子:

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

#include <stdio.h>
//get_max函数的设计
int get_max(int x, int y) {
    return (x>y)?(x):(y);
}
int main() {
    int num1 = 10;
    int num2 = 20;
    int max = get_max(num1, num2);
    printf("max = %d\n", max);
    return 0;
           }

再举个例子:

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

#include <stdio.h>
//实现成函数,但是不能完成任务 
void Swap1(int x, int y) {
    int tmp = 0;
    tmp = x;
    x = y;
    y = tmp;
}
//正确的版本
void Swap2(int *px, int *py) {
    int tmp = 0;
    tmp = *px;
    *px = *py;
    *py = tmp;
}
int main() {
    int num1 = 1;
    int num2 = 2;
    Swap1(num1, num2);
    printf("Swap1::num1 = %d num2 = %d\n", num1, num2);
    Swap2(&num1, &num2);
    printf("Swap2::num1 = %d num2 = %d\n", num1, num2);
    return 0;
}

Swap1 调用的时候,将实参传递给形参

形参其实是实参的一份临时拷贝

对形参的修改不会影响实参

函数的参数

实际参数(实参):

真实传给函数的参数,叫实参。 实参可以是:常量变量表达式函数等。

无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形 参。

形式参数(形参):

形式参数是指函数名后括号中的变量,因为形式参数只有在函数被调用的过程中才实例化(分配内存单元),所以叫形式参数。形式参数当函数调用完成之后就自动销毁了。因此形式参数只在函数中有效。

函数的调用

传值调用

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

传址调用

传址调用是把函数外部创建变量的内存地址传递给函数参数的一种调用函数的方式

这种传参方式可以让函数和函数外边的变量建立起真正的联系,也就是函数内部可以直接操作函数外部的变量

C语言中有布尔类型 (C99引入)

——Bool类型的变量只有两种取值 true和false

#include <stdbool.h>

函数的嵌套调用和链式访问

嵌套调用

#include <stdio.h>
void new_line()
{
    printf("hehe\n");
}
void three_line()
{
    int i = 0;
    for(i=0; i<3; i++)
    {
        new_line();
    }
}
int main() {
    three_line();
return 0; }

函数可以嵌套调用,但是不能嵌套定义。


链式访问

把一个函数的返回值作为另外一个函数的参数。

#include <stdio.h>
#include <string.h>
int main() {
char arr[20] = "hello";
int ret = strlen(strcat(arr,"bit"));//这里介绍一下strlen函数 printf("%d\n", ret);
return 0;
}
#include <stdio.h>
int main() {
printf("%d", printf("%d", printf("%d", 43))); //结果是啥? //注:printf函数的返回值是打印在屏幕上字符的个数 return 0;
}
相关文章
|
1月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
37 3
|
1月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
20 2
|
1月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
66 16
|
1月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
52 1
|
1月前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
61 24
|
1月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
63 23
|
1月前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
68 15
|
2月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
69 9
|
2月前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
64 6
|
2月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
94 10

热门文章

最新文章