C语言中的字符串输入(gets_s、fgets、scanf、fscanf)与相关内存分配知识

0. C语言的内存分配知识

分配内存空间有两种方式:静态内存分配和动态内存分配

0.1 静态内存分配

指的是在编译时确定数组等数据类型的大小,然后由计算机分配好,通常是存在栈上的数据

例如:在声明数组时,需要显示的指明数组的大小

char name[81]; // 一个大小为81bytes的字符数组

0.2 动态内存分配

通俗一点就是向计算机要内存空间,由程序员自由支配,同时程序员也负责释放。执行到相应的程序段时才会去申请内存,而不是像静态内存分配一样提前申请好。

通常使用malloccalloc等内存申请函数

#include<stdlib.h>
void* malloc(size_t size);
// https://www.runoob.com/cprogramming/c-function-calloc.html
void* calloc(size_t nitems, size_t size);
// 申请失败时返回NULL

与申请内存空间相对应的就是释放内存空间,通常使用free函数

//释放之前调用 calloc、malloc 或 realloc 所分配的内存空间
//https://www.runoob.com/cprogramming/c-function-free.html
void free(void *ptr)

1.gets_s()

char *gets_s( char *str, rsize_t n );// (C11 起)

功能:该函数表示从stdin(可以理解为从控制台输入)读取字符直到发现换行符
或出现文件尾EOF

特点:注意该函数会放弃换行符

注意:最多可以写入n-1个字符到str所指向的数组,并在读取完后写入空终止字符' '(也即是第n个字符).

因此可以知道,gets_s函数读入的是一个C风格的字符串

返回值:成功时返回str,失败时返回NULL

用例:

#include <stdio.h>
#include <stdlib.h>

int main() {
    char* str = malloc(10);
    if(str)
        gets_s(str, 10);
    printf("%s
", str);
    return 0;
}
// VS2019中gets_s函数可执行 但是在Linux中无法编译

结果:

原因是:

微软实现了该函数,但是GCC没有实现该函数,详细原因未知。

本人创建的问题链接

2.fgets( )

char *fgets( char *restrict str, int count, FILE *restrict stream );

功能:该函数表示从一个文件流stream中,读取最多count-1个字符,并将读到的字符存储到str所指向的空间中,当函数读取字符遇到EOF或换行符
时结束读取;

特点:如果是读取到换行符
导致的结束读取,该函数将会把换行符
一并存入str所指向的空间中,(这是和gets_s()函数的一大区别,gets_s()函数会抛弃换行符

注意:如果读取数据且没有发生错误,将会在紧随读入的字符数据后写入空终止字符' '

返回值:成功时返回str,失败时返回NULL

用例:

#include <stdio.h>
#include <stdlib.h>

int main()
{
    char *str = malloc(10);
    if (str)
        fgets(str, 10, stdin); // 表示从控制台输入
    printf("%s
", str);
    free(str);
    return 0;
}

结果:

该函数在Linux和Win的VS中均可用

3.scanf()、fscanf()

int scanf( const char *restrict format, ... );// 从stdin格式化的读取数据
int fscanf( FILE *restrict stream, const char *restrict format, ... );// 从某个文件流stream格式化的读取数据

scanf()fscanf()函数确定字符串末尾(也即读取结束)的方法是:

①从非空白开始直至遇到第一个空白字符(包括空行、空格、制表符、换行符)

②若上一次读取键盘输入没有读完,下一次读取会从上次调用函数结束的地方开始

③若指定了读取宽度,则将满足字宽和第一个空白字符**“做或逻辑”**

用例:

#include <stdio.h>
#include <stdlib.h>

int main()
{
    char *str = malloc(10);
    if (str)
        scanf("%s", str);

    printf("%s
", str);
    free(str);
    return 0;
}

结果:

该函数在Linux和Win的VS中均可用