C语言学习系列-->第三弹【浅谈输入和输出函数】

·
2025-11-27 17:04:52

前言 本篇文章,小编浅谈一下C语言中输入和输出函数,由于本人能力有限,部分语言组织可能有问题,(不信)如有谬误,请指正。

一、The basic 知道你急了,但是你别急,这里还是需要说个简单概念,前面也简单说过。 在 C 语言中:

在 C 标准库中,标准输入流输出流分别是 stdin 和 stdout,另外还有标准错误流 stderr。使用 头文件里的 scanf() 函数和 printf() 函数。二、printf()2.1 用法代码语言:javascript复制printf(format,arguement_list); 作用就是向控制台打印数据

这里需要注意的是:字符串中是否包含特殊字符,需要使用相应的转义字符去表示(转义字符之前有讲过,第一弹里面)。

2.2 占位符 举个例子:大学上课前,你的室友帮你占位置,拿一本书占了一个位置,等你来了,你就可以取代那本书啦。

代码语言:javascript复制#include

int main()

{

printf("There are %d apples\n", 3);

printf("%s will come tonight\n", "zhangsan");

printf("%s says it is %d o'clock\n", "lisi", 21);

return 0;

}

// 输出 There are 3 apples

//输出 张三will come tonight

//输出 lisisays it is 21 o'clock2.3其他占位符 •%a :⼗六进制浮点数,字⺟输出为⼩写。

• %A :⼗六进制浮点数,字⺟输出为⼤写。

• %c :字符。

• %d :⼗进制整数。

• %e :使⽤科学计数法的浮点数,指数部分的 e 为⼩写。

• %E :使⽤科学计数法的浮点数,指数部分的 E 为⼤写。

• %i :整数,基本等同于 %d 。

• %f :⼩数(包含 float 类型和 double 类型)。

• %g :6个有效数字的浮点数。整数部分⼀旦超过6位,就会⾃动转为科学计数法,指数部分的 e

为⼩写。

• %G :等同于 %g ,唯⼀的区别是指数部分的 E 为⼤写。

• %hd :⼗进制 short int 类型。

• %ho :⼋进制 short int 类型。

• %hx :⼗六进制 short int 类型。

• %hu :unsigned short int 类型。

• %ld :⼗进制 long int 类型。

• %lo :⼋进制 long int 类型。

• %lx :⼗六进制 long int 类型。

• %lu :unsigned long int 类型。

• %lld :⼗进制 long long int 类型。

• %llo :⼋进制 long long int 类型。

• %llx :⼗六进制 long long int 类型。

• %llu :unsigned long long int 类型。

• %Le :科学计数法表⽰的 long double 类型浮点数。

• %Lf :long double 类型浮点数。

• %a :⼗六进制浮点数,字⺟输出为⼩写。

• %A :⼗六进制浮点数,字⺟输出为⼤写。

• %c :字符。

• %d :⼗进制整数。

• %e :使⽤科学计数法的浮点数,指数部分的 e 为⼩写。

• %E :使⽤科学计数法的浮点数,指数部分的 E 为⼤写。

• %i :整数,基本等同于 %d 。

• %f :⼩数(包含 float 类型和 double 类型)。

• %g :6个有效数字的浮点数。整数部分⼀旦超过6位,就会⾃动转为科学计数法,指数部分的 e

为⼩写。

• %G :等同于 %g ,唯⼀的区别是指数部分的 E 为⼤写。

• %hd :⼗进制 short int 类型。

• %ho :⼋进制 short int 类型。

• %hx :⼗六进制 short int 类型。

• %hu :unsigned short int 类型。

• %ld :⼗进制 long int 类型。

• %lo :⼋进制 long int 类型。

• %lx :⼗六进制 long int 类型。

• %lu :unsigned long int 类型。

• %lld :⼗进制 long long int 类型。

• %llo :⼋进制 long long int 类型。

• %llx :⼗六进制 long long int 类型。

• %llu :unsigned long long int 类型。

• %Le :科学计数法表⽰的 long double 类型浮点数。

• %Lf :long double 类型浮点数。

2.4 输出格式2.4.1 限定宽度 printf() 允许限定占位符的最⼩宽度。

代码语言:javascript复制#include

int main()

{

printf("%5d\n", 123); // 输出为 " 123"

printf("%-5d\n", 123); // 输出为 "123 "

return 0;

} 上⾯⽰例中, %5d 表⽰这个占位符的宽度⾄少为5位。如果不满5位,对应的值的前⾯会添加空格。输出的值默认是右对⻬,即输出内容前⾯会有空格;如果希望改成左对⻬,在输出内容后⾯添加空格,可以在占位符的 % 的后⾯插⼊⼀个 - 号。 对于⼩数,这个限定符会限制所有数字的最⼩显⽰宽度。

代码语言:javascript复制#include

int main()

{

printf("%12f\n", 123.45); // 输出 " 123.450000"

return 0;

}2.4.2 限定小数位数代码语言:javascript复制#include

int main()

{

printf("Number is %.2f\n", 0.5); // 输出 Number is 0.50

return 0;

} 上⾯⽰例中,如果希望⼩数点后⾯输出3位( 0.500 ),占位符就要写成 %.3f 。

2.4.3 结合式写法 以下介绍的写法很少使用

代码语言:javascript复制// 输出为 " 0.50"

#include

int main()

{

printf("%6.2f\n", 0.5);

return 0;

}代码语言:javascript复制#include

int main()

{

printf("%*.*f\n", 6, 2, 0.5);

return 0;

}

// 等同于printf("%6.2f\n", 0.5);2.4.4 输出部分字符串 %s 占位符⽤来输出字符串,默认是全部输出。如果只想输出开头的部分,可以⽤ %.ms 指定输出的⻓度,其中 m 代表⼀个数字,表⽰所要输出的⻓度。

代码语言:javascript复制#include

int main()

{

printf("%.5s\n", "hello world"); // 输出 hello

return 0;

}三、scanf()3.0 出现了问题 我们在vs中使用scanf()函数,会报错,这里,我们需要解决一下

代码语言:javascript复制#define _CRT_SECURE_NO_WARNINGS 1 我们只需要将上述代码放在每一个项目的第一行,就可以解决scanf()函数报错的问题。 针对于这个问题,小编专门写了一篇文章来去解决,有更好更方便更多的方法去解决,请查看《关于vs中scanf()函数报错问题的解决》。

3.1 用法代码语言:javascript复制scanf(format, argument_list); 用于从控制台输入数据,可以读取多种类型的数据,如整数、浮点数、字符、字符串等。

scanf() 输入数据时要求数据格式与 format 字符串中指定的格式匹配,否则会产生错误。

代码语言:javascript复制#include

int main()

{

int score = 0;

printf("请输⼊成绩:");

scanf("%d", &score);

printf("成绩是:%d\n", score);

return 0;

} 你在输⼊的数据之间,有⼀个或多个空格不影响 scanf() 解读数据。另外,使用回车键,将输⼊分成几行,也不影响解读。

3.2 scanf的返回值 scanf() 的返回值是⼀个整数,表⽰成功读取的变量个数。

如果没有读取任何项,或者匹配失败,则返回 0 。如果在成功读取任何数据之前,发⽣了读取错误或者遇到读取到⽂件结尾,则返回常量 EOF。

代码语言:javascript复制#include

int main()

{

int a = 0;

int b = 0;

float f = 0.0f;

int r = scanf("%d %d %f", &a, &b, &f);

printf("a=%d b=%d f=%f\n", a, b, f);

printf("r = %d\n", r);

return 0;

} 如果输⼊2个数后,按 ctrl+z ,提前结束输⼊

在VS环境中按3次 ctrl+z ,才结束了输⼊,我们可以看到r是2,表⽰正确读取了2个数值。如果⼀个数字都不输⼊,直接按3次 ctrl+z ,输出的r是-1,也就是EOF

3.3 占位符 scanf() 常⽤的占位符如下,与 printf() 的占位符基本⼀致。 • %c :字符。

• %d :整数。

• %f : float 类型浮点数。

• %lf : double 类型浮点数。

• %Lf : long double 类型浮点数。

• %s :字符串。

• %[] :在⽅括号中指定⼀组匹配的字符(⽐如 %0-9 ),遇到不在集合之中的字符,匹配将会停⽌。 特别说⼀下占位符 %s ,它其实不能简单地等同于字符串。它的规则是,从当前第⼀个⾮空⽩字符开始读起,直到遇到空⽩字符(即空格、换⾏符、制表符等)为⽌。

所以⽆法⽤来读取多个单词,除⾮多个 %s ⼀起使⽤。 scanf() 将字符串读⼊字符数组时,不会检测字符串是否超过了数组⻓度。所以,储存字符串时,很可能会超过数组的边界,导致预想不到的结果。为了防⽌这种情况,使⽤ %s 占位符时,应该指定读⼊字符串的最⻓⻓度,即写成 %ms ,其中的 m 是⼀个整数,表⽰读取字符串的最⼤⻓度,后⾯的字符将被丢弃。

代码语言:javascript复制#include

int main()

{

char name[11];

scanf("%10s", name);

return 0;

} name 是⼀个⻓度为11的字符数组, scanf() 的占位符 %10s 表⽰最多读取⽤⼾输⼊的10个字符,后⾯的字符将被丢弃,这样就不会有数组溢出的⻛险了。

3.3.4 赋值忽略符代码语言:javascript复制#include

int main()

{

int year = 0;

int month = 0;

int day = 0;

scanf("%d-%d-%d", &year, &month, &day);

return 0;

} 上⾯⽰例中,如果⽤⼾输⼊ 2020-01-01 ,就会正确解读出年、⽉、⽇。问题是⽤⼾可能输⼊其他格式,⽐如 2020/01/01 ,这种情况下, scanf() 解析数据就会失败。 为了避免这种情况, scanf() 提供了⼀个赋值忽略符(assignment suppression character) * 。只要把 * 加在任何占位符的百分号后⾯,该占位符就不会返回值,解析后将被丢弃。

代码语言:javascript复制#include

int main()

{

int year = 0;

int month = 0;

int day = 0;

scanf("%d%*c%d%*c%d", &year, &month, &day);

return 0;

} 上⾯⽰例中, %*c 就是在占位符的百分号后⾯,加⼊了赋值忽略符 * ,表⽰这个占位符没有对应的变量,解读后不必返回。

总结 本节主要是对输入和输出函数进行简单的讲解,要想有更深入的了解,可以去搜索。当然,在目前学习中,掌握这些其实足够了。

https://legacy.cplusplus.com/reference/cstdio/scanf/?kw=scanf

https://legacy.cplusplus.com/reference/cstdio/printf/?kw=printf