当前位置: 首页 > news >正文

C语言第一章数据类型和变量(下)

九.scanf和printf函数的介绍

1.printf函数

(1)printf函数的基本用法

        printf中print是打印的意思,f是format的首字母,意思是格式化。所以printf的意思就是按照一定的格式把一个东西打印出来。

如果需要在屏幕上打印出“Hello World!”,那么请看下方代码:

# include <stdio.h>
int main()
{
printf("Hello World!");
return 0;
}

因为printf是C语言中的库函数,所以需要包含头文件<stdio.h>。上述代码运行的结果为:Hello World!

(2)占位符

printf函数可以在输出内容中指定占位符,所谓占位符就是会被文本替换掉的符号。举个例子:

#include <stdio.h>
int main()
{
printf("%d",100);
return 0;
}

上述代码打印出来的结果为100

#include<stdio.h>
int main()
{
printf("我的年龄是%d岁",100);
return 0;
}

上述代码打印的结果为:我的年龄是100岁

综上两个例子,可以看出占位符在打印的时候会被指定的数据替换掉。

占位符的第⼀个字符⼀律为百分号 % ,第二个字符表示占位符的类型, %d 表示这里代入的值必须是⼀个整数。

除了%d为整型的占位符以外,还有很多其它类型的占位符。下面介绍字符串的占位符%s。

#include <stdio.h>
int main()
{printf("%s暑假作业没写完", "张三");return 0;
}

上述代码将张三替换掉%s,最终打印的结果是:张三暑假作业没写完

上面是对占位符%d和%s的介绍,下面介绍printf的参数。

(3)printf的参数

printf的参数个数与占位符有关,如果占位符有N个,那么该printf函数就有N+1个参数,下面将举例说明这一问题。

#include <stdio.h>
int main()
{
printf("%s的学号是%d","王华",211);
return 0;
}

上述代码打印的结果为:王华的学号是211。一共有两个占位符。分别为%d和%s,所以该printf函数的参数有三个,分别是%s%d%s的学号是%d。

注意:如果参数的个数少于占位符的个数,那么printf函数就会打印内存中的任意值。

(4)占位符详情

%a:十六进制浮点数,字母输出为小写。

%A:十六进制浮点数,字母输出为大写。

%c:字符。

%d :十进制整数。

%e: 使用科学计数法的浮点数,指数部分的 e 为小写。

%E :使用科学计数法的浮点数,指数部分的 E 为大写。

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

%f :小数,也就是浮点数。

%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类型浮点数。

%n :已输出的字符串数量。该占位符本⾝不输出,只将值存储在指定变量之中。  

%o :八进制整数。

%p :指针(用来打印地址)。 

%s :字符串。

%u :无符号整数(unsigned int)。

%x :⼗六进制整数。 

%zd : size_t 类型。

%% :输出⼀个百分号。

(5)输出格式

<1>限定宽度

printf函数允许限定占位符的最小宽度

#include <stdio.h>
int main()
{printf("%5d\n", 123); // 输出为 " 123" return 0;
}

上面示例中, %5d 表⽰这个占位符的宽度至少为5位。如果不满5位,对应的值的前面会添加空格。 输出的值默认是右对齐,即输出内容前面会有空格;如果希望改成左对齐,在输出内容后面添加空格,可以在占位符的 % 的后面插入一个 负号。

例如,下面这个代码:

#include <stdio.h>
int main()
{printf("%-5d\n", 123); // 输出为 "123 " return 0;
}

上面是整数的例子,下面举一个小数的例子:

#include <stdio.h>
int main()
{printf("%12f\n", 123.45);return 0;
}

 输出的结果为:"   123.450000"。 %12f 表示输出的浮点数最少要占据12位。由于小数的默认显示精度是小数点后6位, 所以 123.45 输出结果的头部会添加2个空格。

<2>显示正负号 

默认情况下,printf函数打印出来的数字不会显示正负号,为了能让printf函数显示出正负号,请看下面的代码实践:

#include <stdio.h>
int main()
{printf("%+d\n", 12); // 输出 +12 printf("%+d\n", -12); // 输出 -12 return 0;
}

如上所示:要想让printf函数打印出正负号,只需要在占位符前面加上正负号即可。

<3>限定小数位数

输出小数时,有时希望限定小数的位数。举例来说,希望小数点后⾯只保留两位,占位符可以写 成 %.2f 。请看下面的代码实践:

// 输出 Number is 0.50 
#include <stdio.h>
int main()
{printf("Number is %.2f\n", 0.5);return 0;
}

上述代码中的%.2f就可以让0.5保留两位小数。

这种写法可以和限定宽度的同时使用,比如下面代码:

// 输出为 " 0.50" 
#include <stdio.h>
int main()
{printf("%6.2f\n", 0.5);return 0;
}

上述代码就可以同时限定小数的宽度,并且还能限定小数的位数。

最小宽度和小数位数这两个限定值,都可以用 * 代替,通过 printf函数的参数传入。请看下面代码实践:

#include <stdio.h>
int main()
{printf("%*.*f\n", 6, 2, 0.5);return 0;
}
// 等同于printf("%6.2f\n", 0.5);

这样比起上一种代码的写法好处是:可以随时变化要限定的宽度和小数的位数。

<4>输出部分字符串

%s 占位符用来输出字符串,默认是全部输出。如果只想输出开头的部分,可以用 %.m s 指定输出 的长度,其中 m 是一个数字,表示所要输出的长度。

// 输出 hello 
#include <stdio.h>
int main()
{printf("%.5s\n", "hello world");return 0;
}

上述代码只会打印字符串的前5个字母:hello

2.scanf函数

(1)函数介绍

当我们有了变量,我们需要给变量输入值就可以使用 scanf 函数,如果需要将变量的值输出在屏幕上 的时候可以使用 printf 函数,下面看⼀个例子:

#include <stdio.h>
int main()
{int score = 0;printf("请输⼊成绩:");scanf("%d", &score);printf("成绩是:%d\n", score);return 0;
}

 上述代码会先打印出:请输入成绩:,然后黄标就会闪烁,等待程序员输入一个值。如果你输入的是一个整型值100,那么接下来就会打印:成绩是:100

(2)基本用法

        scanf函数用于读取用户的键盘输入。 程序运行到这个语句时,会停下来,等待用户从键盘输入。 用户输入数据、按下回车键后, scanf函数就会处理用户的输入,将其存入变量。 它的原型定义在头文件stdio.h 。 scanf函数的语法跟 printf函数类似。

scanf("%d",&i);

        scanf里第一个参数是格式字符串,会放置占位符,与printf函数的占位符一致。作用是告诉编译器键盘输入数据的类型。这是因为C语言中,数据有着不同的类型,C语言要提前明确不同的类型,才能处理数据。

        scanf的其余参数就是存放用户输入变量的地址,有多少个占位符就有多少个变量地址。

上述例子中,%d表示用户输入的是一个整型变量;&i表示将用户输入的值存放在变量i的地址里。(其中&操作符是取地址操作符,作用是取出变量对应的地址。)

注意:变量前面必须加上 & 运算符(指针变量除外),因为 scanf() 传递的不是值,而是地址, 即把变量 i 的地址指向用户输入的值。 如果这里的变量是指针变量(比如字符串变量),那就不用加 & 运算符。 

下面给一个scanf函数读取多个变量输入数据的例子:

scanf ("%d %d %f %f ",&i ,&j ,&x ,&y );

上述代码%d%d%f%f指的是有两个整型两个浮点型的值待输入。比如: 1  -20  3.4  -4.0e3 。这四个值依次放入 i 、 j 、 x 、 y 四个变量。 scanf函数处理数值占位符时,会自动过滤空白字符,包括空格、制表符、换行符等。 所以,用户输入的数据之间,有一个或多个空格不影响 scanf函数的解读数据。另外,用户使用回车键,将输入分成几行,也不影响解读。

1

-20

3.4

-4.0e3

如上方情况输入时,得到的结果与一行输入是完全一样的。每次按下回车键以后, scanf函数就会开始解读,如果第一行匹配第一个占位符,那么下次按下回车键时,就会从第⼆个占位符开始解读。 scanf函数处理用户输入的原理是:用户

输入先放入缓存,等到按下回车键后,按照占位符对缓存进行解读。 解读用户输入时,会从上一次解读遗留的第一个字符开始,直到读完缓存,或者遇到第一个不符合占位符条件的字符为止。下面将举例解释:

#include <stdio.h>
int main()
{int x;float y;// ⽤⼾输⼊ " -13.45e12# 0" scanf("%d", &x);printf("%d\n", x);scanf("%f", &y);printf("%f\n", y);return 0;
}

上面示例中, scanf函数读取用户输入时,%d 占位符会忽略刚开始的空格,从 - 处开始获取数据,读取到 -13 停下来,因为后面的数据不属于整数的有效字符。这就是说占位符 %d 会读到 -13 。 第二次调用 scanf函数时,就会从上一次停止解读的地方,继续往下读取。这⼀次读取的首字符 是“ . ”,由于对应的占位符是 %f ,会读取到 .45e12 ,这是采用科学计数法的浮点数格式。后面的 # 不属于浮点数的有效字符,所以会停在这里。因为scanf可以同时处理多个用户输入,所以代码也可以是这样:

#include <stdio.h>
int main()
{int x;float y;// ⽤⼾输⼊ " -13.45e12# 0" scanf("%d%f", &x, &y);return 0;
}

(3)scanf函数的返回值 

        scanf() 的返回值是一个整数,表示成功读取的变量个数。 如果没有读取任何项,或者占位符格式和数据格式匹配失败,则返回 0 。 如果在成功读取任何数据之前,发生了读取错误或者遇到读取到文件结尾,则返回常量 EOF (-1)。 EOF : end of file意思是文件结尾标志

#include <stdio.h>
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;
}

 输入输出测试:

如果输入1 2 两个值后,按Ctrl+z提前结束输入则:

我们可以看到f没有获取到用户输入的数据,所以默认为初始化的值0,且r=2也证实了scanf的返回值是正确读取变量数据的个数。下面是没有数据输入,直接Ctrl+Z提前结束输入的情况。

如果一个数据也不输入,则都被默认为初始化的值0,因为scanf函数一个数据也没有读取,所以它的返回值是-1也就是EOF。

(4)占位符

scanf函数常用的占位符如下(与 printf函数的占位符基本⼀致。):

%c :字符。

%d :整数。

%f : float 类型浮点数。

%lf : double 类型浮点数。

%Lf : long double 类型浮点数。

%s :字符串。

%[ ] :在方括号中指定⼀组匹配的字符(比如 %[0-9] ),遇到不在集合之中的字符,读取将会停止。

        上面所有占位符之中,除了 %c 以外,其他的占位符都会自动忽略开始的空白字符。%c 不忽略空白字符,总是返回当前第⼀个字符,无论该字符是否为空格。 如果要强制跳过字符前的空白字符,可以写成 scanf("   %c", &ch) ,即%c 前加上⼀个空格,表示跳过零个或多个空白字符。

        为什么scanf函数不适合读取包含空格的字符串???

         下面要特别说⼀下占位符 %s ,它其实不能简单地等同于字符串。它的规则是,从当前第⼀个非空白字符开始读起,直到遇到空白字符(即空格、换行符、制表符等)为止。 因为 %s 不会包含空白字符,所以无法用来读取多个单词,除非多个%s ⼀起使用。这也意味着:scanf函数不适合读取可能包含空格的字符串,比如书名或歌曲名。另外,scanf函数遇到 %s 占位符时,会在字符串变量末尾存储⼀个空字符 ‘\0’ 。 scanf函数将字符串读入字符数组时,不会检测字符串是否超过了数组长度。所以,储存字符串时, 很可能会超过数组的边界,导致预想不到的结果。为了防止这种情况,使用%s 占位符时,应该指定读入字符串的最长长度,即写成 %[m]s ,其中的 [m] 是⼀个整数,表示读取字符串的最大长度,后面的字符将被丢弃。

#include <stdio.h>
int main()
{char name[11];scanf("%10s", name);return 0;
}

上述代码中,定义了一个名为name长度为11的字符型数组 ,scanf函数的占位符 %10s 表示:最多读取用户输入的10个字符,后面的字符将被丢弃,这样就不会有数组溢出的风险了。

(4)赋值忽略符

有时,用户的输入可能不符合预定的格式。我们需要把用户多余输入的无关紧要的字符忽略掉,这时候我们就会用到赋值忽略符。

举一个简单的例子,每个人输入年月日都有不同的习惯,有的喜欢这样2025.4.2有的喜欢这样2025/4/2或者这样2025-4-2:

#include <stdio.h>
int main()
{int year = 0;int month = 0;int day = 0;scanf("%d-%d-%d", &year, &month, &day);printf("%d %d %d\n", year, month, day);return 0;
}

上面示例中,如果用户输入2020-01-01,就会正确解读出年、月、日。问题是用户可能输入其他 格式,比如 2020/01/01 ,这种情况下, scanf函数解析数据就会失败。为了避免这种情况的发生, scanf函数提供了一

个赋值忽略符:“* ”。 只要把 * 加在任何占位符的百分号后面,该占位符就不会返回值,解析后将被丢弃。就像下面那样:

#include <stdio.h>
int main()
{int year = 0;int month = 0;int day = 0;scanf("%d%*c%d%*c%d", &year, &month, &day);return 0;
}

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

http://www.dtcms.com/a/276122.html

相关文章:

  • Java 大视界:基于 Java 的大数据可视化在智慧城市能源消耗动态监测与优化决策中的应用(2025 实战全景)
  • 视频分析应用的搭建
  • 【Linux-云原生-笔记】Apache相关
  • NE综合实验2:RIP与OSPF动态路由优化配置、FTP/TELNET服务部署及精细化访问控制
  • Java反射与注解
  • 树形动态规划详解
  • 大数据时代UI前端的智能化服务升级:基于用户情境的主动服务设计
  • 【PycharmPyqt designer桌面程序设计】
  • 【学习新知识】用 Clang 提取函数体 + 构建代码知识库 + AI 问答系统
  • GD32 CAN1和TIMER0同时开启问题
  • 《通信原理》学习笔记——第一章
  • 细谈kotlin中缀表达式
  • H2在springboot的单元测试中的应用
  • skywalking镜像应用springboot的例子
  • try-catch-finally可能输出的答案?
  • Docker-镜像构建原因
  • C语言基础教程--从入门到精通
  • Spring Boot整合MyBatis+MySQL+Redis单表CRUD教程
  • STM32中的RTC(实时时钟)详解
  • R 语言绘制 10 种精美火山图:转录组差异基因可视化
  • JavaScript 常见10种设计模式
  • 码头智能哨兵:AI入侵检测系统如何终结废钢盗窃困局
  • Redis专题总结
  • MyBatis实现一对多,多对一,多对多查询
  • Golang操作MySQL json字段优雅写法
  • CPU缓存一致性协议:深入解析MESI协议与多核并发设计
  • HTML/JOSN复习总结
  • 7. JVM类加载器与双亲委派模型
  • PyQt5 — QTimeEdit 学习笔记
  • Java中的wait和notify、Condition接口的使用