主要介绍C语言中比较基础,也比较常用的两个函数,printf()与scanf()。
一、printf()函数
(1)printf()函数基础介绍
print:打印 format:格式
printf:按照一定格式打印,即格式化输出,将参数文本(双引号内的内容)输出至屏幕上
语法:printf(),为一种库函数,使用时需头文件<stdio.h>
#include <stdio.h> // printf()函数使用所需的头文件
int main(void)
{
// 双引号内的内容“hello World”会输出至屏幕上
printf("Hello World\n"); // \n,换行符,打印完后光标停留至下一行开头
return 0;
}
注:
- 使用printf()函数时打印的内容一定要放在双引号(英文符号)中间,否则打印时屏幕中会无内容
- printf()函数不会自动换行,如需换行,可在打印内容末尾加入换行符 \n
(2)占位符
“占位符”就是先占住一个固定的位置,再往里面添加内容的符号,即所占的位置可以用其它值代入。
printf()函数中可以在输出⽂本中指定占位符。其使用格式:% + 数据类型,如:%d,表示为十进制整数占据了一个位置。
常用占位符列举:
- %d----十进制整数、%s----字符串、%c----字符、%f----小数(float和double类型)
- %o----八进制、%x----十六进制、%u----无符号十进制数据
- %ld----十进制 long int类型
- %a----⼗六进制浮点数,字⺟输出为⼩写、%A----⼗六进制浮点数,字⺟输出为大写
- %e----使⽤科学计数法的浮点数,指数部分的e小写、%E----使⽤科学计数法的浮点数,指数部分的E大写
#include <stdio.h>
int main()
{
printf("Here are %d car\n", 1); // 1 替换 %d
printf("Here are %s cars\n", "two"); // two 替换 %s
printf("Here are %c cars\n", 'a'); // a 替换 %c
// 多个占位符一起使用时,逗号后的内容会依次替换,类型一一对应
// 'I' 1 "two" 依次替换 %c、%d、%s
printf("%c have %d cars instead of %s\n", 'I', 1, "two" );
return 0;
}
运行结果:
(3)输出格式化定制
printf()可以定制占位符的输出格式。如限定最小宽度、显示正负号、限定小数位数、输出部分字符串等。
- %5d,将整型数据按照宽度至少为5进行输出,且采取右对齐方式
- %12f,将浮点型数据(默认保留6位小数)按照宽度至少为12进行输出,且采取右对齐方式
- %.3f,点+数字的浮点型数据,表示保留三位小数
- %5.2f,将浮点型数据按照宽度至少为5进行输出,且保留两位小数,默认右对齐
- %.5s,点+数字的字符串类型,表示输出字符串前5个字符
- %+d,输出正号;%-d,输出负号
#include <stdio.h>
int main()
{
// %5d,表示占位符至少5位宽度,
// 若不足,则前面加空格,默认为右对齐,要使其左对齐,则为 %-5d
printf("%5d\n", 123); // 输出 " 123",右对齐
printf("%-5d\n", 123); // 输出 "123 ",左对齐
// %12f,表示输出的浮点数最少要占据12位
// 小数的默认显示精度是小数点后6位
printf("%12f\n", 123.45); // 输出 " 123.450000"
// 默认情况下,printf() 不对正数显示 + 号,只对负数显示 - 号
// 要显示 + 号,只需 %+d
printf("%+d\n", 123); // 输出 "+123"
printf("%+d\n", -123); // 输出 "-123"
// %.3f,表示保留三位小数
printf("%.3f\n", 10.5796); // 输出 "10.580"
// %5.2f,表示输出的浮点数最小宽度为5,小数位数为2
printf("%5.2f\n", 1.5); // 输出 " 1.50"
// 最小宽度和小数位数这两个限定值,都可以⽤*代替
// 通过 printf() 的参数传入
// 第一个*为5,第二个*为2,表示输出字符串最小宽度为5,小数位数为2
printf("%*.*f\n", 5,2,1.5); // 输出 " 1.50"
// %s,默认输出全部字符串,可用%.[m]s指定输出长度,[m]为数字
// %.5s,表示输出字符串前5个字符
printf("%.5s\n", "hello world"); // 输出"hello"
return 0;
}
运行结果:
二、scanf()函数
scanf()是C语言标准库中的一个输入函数,与printf()函数一样,使用scanf()函数时要加上头文件<stdio.h>。
注:标准输⼊⼀般指的就是键盘,标准输出⼀般指的就是屏幕
(1)scanf()函数基础介绍
scanf():按照一定格式打印,即格式化输入,按指定格式读取用户的输入到指定的变量之中。
执行过程:用户输入数据、按下回车键后,scanf()就会处理用户的输入,将其存⼊指定变量。
语法: scanf(“<格式化字符串>”,<地址表>),必须提前知道⽤⼾输⼊的数据类型,才能处理数
据。如: scanf(“%d”,&score),从键盘中读取一个整型数据,并存入至变量score中。
#define _CRT_SECURE_NO_WARNINGS 1 // 解决scanf()的报错问题
#include <stdio.h> // printf()、sacnf()函数使用所需的头文件
int main()
{
int score = 0;
printf("请输入你的成绩:");
//注意:变量前⾯必须加上 & 运算符(指针变量除外),因为scanf() 传递的不是值,而是地址
// 即将变量score 的地址指向用户输入的值
// 如果这里的变量是指针变量(比如字符串变量),那就不用加 & 运算符
scanf("%d",&score); // 从键盘中读取一个整型数据,并存入至变量score中
printf("成绩是:%d",score);
return 0;
}
运行结果:
关于scanf在Visual Studio中报错问题的解决:
如上图,在Visual Studio环境中,认为scanf函数是不安全的,建议使用scanf_s来代替。要想使这个错误描述失效,请使用 _CRT_SECURE_NO_WARNINGS。
原因:scanf_s是Visual Studio这个集成环境(IDE)所提供的,但在C语言中scanf_s不是其标准库的函数,也就是scanf_s这个函数只能在Visual Studio所提供的环境下使用,若换个环境,则不可使用,编译器是编译不了的,降低了代码的可移植性。
解决办法:
方法一:在当前代码第一行中加上 #define _CRT_SECURE_NO_WARNINGS 1
#define _CRT_SECURE_NO_WARNINGS 1
方法二:修改Visual Studio 2022软件安装路径下的 newc++file.cpp 文件,在其中加入 #define _CRT_SECURE_NO_WARNINGS 1 这句话即可,每次新建 .c或 .c++文件时会自动帮我们在第一行加入这句话,就不用每次使用scanf()函数时手动加入。
newc++file.cpp 文件修改步骤:使用Everything软件直接搜索newc++file.cpp并选中该文件---->右键选择打开路径---->用记事本打开newc++file.cpp文件,加入 #define _CRT_SECURE_NO_WARNINGS 1 这句话即可。
Everything软件下载地址:https://www.voidtools.com/downloads/
方法三:在当前代码中加入:#pragma warning(disable:4996),使该警告失效也可
#pragma warning(disable:4996)
注:
- scanf()处理数值占位符时,会自动过滤空⽩字符,包括空格、制表符、换⾏符等
- scanf()处理用户输入的原理是:用户的输入先放入缓存,等到按下回⻋键后,按照占位符对缓存进行解读
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int x;
float y;
scanf("%d%f", &x, &y); // 用户输入 " -13.45e12# 0"
/**
scanf()读取用户输入时,%d 占位符会忽略起⾸的空格,从-处开始获取数据,读取到-13 停下来,
因为后⾯的.不属于整数的有效字符。这就是说,占位符%d 会读到-13 。
第⼆次调⽤scanf()时,就会从上⼀次停⽌解读的地⽅,继续往下读取。这⼀次读取的⾸字符
是. ,由于对应的占位符是%f,会读取到.45e12 ,这是采⽤科学计数法的浮点数格式。后⾯的
# 不属于浮点数的有效字符,所以会停在这⾥。
*/
printf("%d %f", x, y); // 输出结果:-13 449999994880.000000
// 科学计数法:1.5e3 等价于 1.5*10^3=1500
return 0;
}
(2)scanf()函数的返回值
scanf()函数的返回值是⼀个整数,表示成功读取的变量个数,若没有读取任何项,或者匹配失败,则返回0;若在成功读取任何数据之前,发生读取错误或者遇到读取到文件结尾,则返回常量EOF。
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h> // printf()、sacnf()函数使用所需的头文件
int main()
{
// 创建三个变量a,b,f并初始化
int a = 0;
int b = 0;
float f = 0.0f;
// 从键盘中依次读取输入的数据,并依次存放至指定变量中后
// 将成功读取到的数据个数返回给变量 d
int d = scanf("%d %d %f", &a, &b, &f); // 依次输入 1 2 1.5
printf("a=%d b=%d f=%f\n", a, b, f);
printf("d = %d\n", d); // scanf成功返回个数为 3
return 0;
}
运行结果:
(3)占位符与赋值忽略符
scanf()函数常用占位符与printf()函数的占位符基本一致。
数据类型 | 格式符 | 使用方式 |
---|---|---|
int | %d | scanf(“%d”,&score) |
float | %f | scanf(“%f”,&score) |
double | %lf | scanf(“%lf”,&score) |
long double | %Lf | scanf(“%Lf”,&score) |
char | %c | scanf(“%c”,&score) |
字符串 | %s | scanf(“%s”,&score) |
上⾯所有占位符之中,除了%c 以外,都会自动忽略一开始的空⽩字符。%c 不忽略空白字符,总是返回当前第⼀个字符,⽆论该字符是否为空格。若要强制跳过字符前的空⽩字符,可在%c前加一个空格,表示跳过零个或多个空⽩字符。
scanf(" %c",&ch);
占位符%s的使用注意事项:
- scanf()函数读取%s类型的数据时,从当前第⼀个非空白字符开始读起,直到遇到空白字符(即空格、换⾏符、制表符等)为止,就会停下来。也就是说,scanf()函数不适合读取可能包含空格的字符串。另外,scanf()函数会在字符串变量末尾存储⼀个空字符 \0 。
- scanf()函数将字符串读⼊字符数组时,不会检测字符串是否超过了数组长度。所以,储存字符串时,很可能会超过数组的边界,导致预想不到的结果。为了防⽌这种情况,使用%s 占位符时,应该指定读入字符串的最长长度,即写成%[m]s ,其中的[m] 是⼀个整数,表示读取字符串的最大长度,后面的字符将被丢弃。
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
char name[11];
// 读取输入字符串最大长度为10字节,一个汉字2字节
scanf("%10s", &name ); // 输入:张胜利哈哈哈
printf("%s\n", name); // 输出结果:张胜利哈哈
return 0;
}
运行结果:
赋值忽略符:*
当用户不按指定格式输入时,输出的数据就会存在问题。如:指定输入格式为 xxxx-xx-xx ,结果用户 2023/11/26 这样输入,就会导致输出结果为 2023 0 0 。
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int year = 0;
int month = 0;
int day = 0;
scanf("%d-%d-%d", &year, &month, &day); // 指定输入格式为:2023-11-26
printf("%d %d %d", year, month, day); // 若不按指定格式输入,则输出结果为:2023 0 0
return 0;
}
解决方法:为了避免这种情况,scanf()提供了⼀个赋值忽略符 * ,只要把 * 加在任何占位符的百分号后面,该占位符就不会返回值,解析后将被丢弃。
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int year = 0;
int month = 0;
int day = 0;
scanf("%d%*c%d%*c%d", &year, &month, &day); // %*c,表示该类型占位符使用后会被丢弃
printf("%d %d %d\n", year, month, day);
return 0;
}
运行结果: