printf输出例子_输出printf语句

printf输出例子_输出printf语句C语言printf() 详解之终极无惑1.printf() 简介printf() 是 C 语言标准库函数,用于将格式化后的字符串输出到标准输出。标准输出,即标准输出文件,对应终端的屏幕。printf() 申明于头文件 stdio.h。函数原型&

C语言printf() 详解之终极无惑
  1.printf() 简介

  printf() 是 C 语言标准库函数,用于将格式化后的字符串输出到标准输出。标准输出,即标准输出文件,对应终端的屏幕。printf() 申明于头文件 stdio.h。

  函数原型:

  返回值:

  正确返回输出的字符总数,错误返回负值。与此同时,输入输出流错误标志将被置值,可由指示器函数 ferror(FILE *stream) 来检查输入输出流的错误标志,如果 ferror() 返回一个非零值,表示出错。

  调用格式:

  格式化字符串包含三种对象,分别为:

  (1)字符串常量;

  (2)格式控制字符串;

  (3)转义字符。

  字符串常量原样输出,在显示中起提示作用。输出表列中给出了各个输出项,要求格式控制字符串和各输出项在数量和类型上应该一一对应。其中格式控制字符串是以 % 开头的字符串,在 % 后面跟有各种格式控制符,以说明输出数据的类型、宽度、精度等。

  2.格式控制字符串详解

  printf() 的格式控制字符串组成如下:

  分别为:

  2.1 类型(type)

  首先说明类型,因为类型是格式控制字符串的重中之重,是必不可少的组成部分,其它的选项都是可选的。type 用于规定输出数据的类型,含义如下:
字符对应数据类型含义示例d/iint输出十进制有符号 32bits 整数,i 是老式写法输出123ounsigned int无符号8进制(octal)整数(不输出前缀0)输出0173uunsigned int无符号10进制整数输出123x/Xunsigned int无符号16进制整数,x对应的是abcdef,X对应的是ABCDEF(不输出前缀0x)输出0x7b 0x7Bf/lffloat(double)单精度浮点数用f,双精度浮点数用lf(printf可混用,但scanf不能混用)输出0.000000123 0.000000123。注意指定精度,否则printf默认精确到小数点后六位Ffloat(double)与f格式相同,只不过 infinity 和 nan 输出为大写形式。例如输出结果为e/Efloat(double)科学计数法,使用指数(Exponent)表示浮点数,此处"e"的大小写代表在输出时“e”的大小写输出1.230000e-07 1.230000E-07gfloat(double)根据数值的长度,选择以最短的方式输出,%f或%e输出1.23e-07 0.123Gfloat(double)根据数值的长度,选择以最短的方式输出,%f或%E输出1.23E-07 0.123cchar字符型。可以把输入的数字按照ASCII码相应转换为对应的字符输出Aschar*字符串。输出字符串中的字符直至字符串中的空字符(字符串以空字符’0‘结尾)输出:测试testSwchar_t*宽字符串。输出字符串中的字符直至字符串中的空字符(宽字符串以两个空字符’0‘结尾)

  输出:测试test | | p | void* | 以16进制形式输出指针 | 输出:0x000000013FF73350 | | n | int* | 什么也不输出。%n对应的参数是一个指向signed int的指针,在此之前输出的字符数将存储到指针所指的位置 |

  输出:lvlvnum:4 | | % | 字符% | 输出字符‘%’(百分号)本身 | 输出:% | | m | 无 | 打印errno值对应的出错内容 |  | | a/A | float(double) | 十六进制p计数法输出浮点数,a为小写,A为大写 | 输出:0x1.e4ccccccccccdp+3 0X1.E4CCCCCCCCCCDP+3 |

  注意:

  (1)使用 printf() 输出宽字符时,需要使用 setlocale 指定本地化信息并同时指明当前代码的编码方式。除了使用 %S,还可以使用 %ls。

  (2)printf() 输出 bool 类型无专用类型标识符,实际输出时按照整型 0 或 1 输出布尔值。

  (3)%a 和 %A 是 C99 引入的格式化类型,采用十六进制 p 计数法输出浮点数。p 计数法类似 E 科学计数法,但有所不同。数以 0x 开头,然后是 16 进制浮点数部分,接着是 p 后面是以 2 为底的阶码。以上面输出的 15.15 为例,推算输出结果。15.15 转换成二进制为,因为二进制表示数值的离散特点,计算机对于小数有时是不能精确表示的,比如 0.5 可以精确表示为 0. 1 2 0.1_2 0.12,而 0.15 却不能精确表示。将15.15 对应的二进制右移三位,为转换对应的十六进制就是0x1.e4ccccccccccd,注意舍入时向高位进了1位。由于右移三位,所以二进制阶码是 3。最后的结果就是 0x1.e4ccccccccccdp+3。

  (4)格式控制字符串除了指明输出的数据类型,还可以包含一些其它的可选的格式说明,依序有 flags, width, .precision and length。下面一一讲解。

  2.2 标志(flags)

  flags 规定输出样式,取值和含义如下:
字符名称说明-减号结果左对齐,右边填空格。默认是右对齐,左边填空格。+加号输出符号(正号或负号)space空格输出值为正时加上空格,为负时加上负号#井号type是o、x、X时,增加前缀0、0x、0X。

  type是a、A、e、E、f、g、G时,一定使用小数点。默认的,如果使用.0控制不输出小数部分,则不输出小数点。

  type是g、G时,尾部的0保留。| | 0 | 数字零 | 将输出的前面补上0,直到占满指定列宽为止(不可以搭配使用“-”) |

  示例:

  输出结果为:

  printf输出例子_输出printf语句

  2.3 输出最小宽度(width)

  用十进制整数来表示输出的最少位数。若实际位数多于指定的宽度,则按实际位数输出,若实际位数少于定义的宽度则补以空格或0。width的可能取值如下:
width描述示例数值十进制整数输出:001000*星号。不显示指明输出最小宽度,而是以星号代替,在printf的输出参数列表中给出输出:001000

  2.4 精度(.precision)

  精度格式符以“.”开头,后跟十进制整数。可取值如下:
.precision描述.数值十进制整数。

  (1)对于整型(d,i,o,u,x,X),precision表示输出的最小的数字个数,不足补前导零,超过不截断。

  (2)对于浮点型(a, A, e, E, f ),precision表示小数点后数值位数,默认为六位,不足补后置0,超过则截断。

  (3)对于类型说明符g或G,表示可输出的最大有效数字。

  (4)对于字符串(s),precision表示最大可输出字符数,不足正常输出,超过则截断。

  precision不显示指定,则默认为0 | | .* | 以星号代替数值,类似于width中的*,在输出参数列表中指定精度。|

  示例:

  输出结果:

  注意: 在对浮点数和整数截断时,存在四舍五入。

  2.5 类型长度(length)

  类型长度指明待输出数据的长度。因为相同类型可以有不同的长度,比如整型有 char(8bits)、short int(16bits),int(32bits)和 long int(64bits),浮点型有 32bits 的单精度 float 和 64bits 的双精度 double。为了指明同一类型的不同长度,于是乎,类型长度(length)应运而生,成为格式控制字符串的一部分。

  因为 Markdown 表格不支持单元格合并,背景颜色等样式,所以直接引用C++ reference.printf http://www.cplusplus.com/reference/cstdio/printf/?kw=printf)的表格。

  printf输出例子_输出printf语句

  注意: 黄色背景行标识的类型长度说明符和相应的数据类型是C99引入的。

  示例代码:

  输出结果:

  注意:

  long int 到底是 32bits 还是 64bits 跟生成的程序是 32bits 还是 64bits 一一对应,如果使用 g++ 编译程序的话,可通过或选项分别生成 32bits 和 64bits 的程序。因本人测试代码编译生成的是 64bits 的程序,所以 long int 也就是 64btis。

  3.转义字符

  转义字符在字符串中会被自动转换为相应操作命令。printf() 使用的常见转义字符如下:
转义字符意义a警报(响铃)符b回退符f换页符

  换行符
回车符 横向制表符v纵向制表符\反斜杠"双引号

  4.关于 printf 缓冲

  在 printf 的实现中,在调用 write 之前先写入 IO 缓冲区,这是一个用户空间的缓冲。系统调用是软中断,频繁调用,需要频繁陷入内核态,这样的效率不是很高,而 printf 实际是向用户空间的 IO 缓冲写,在满足条件的情况下才会调用 write 系统调用,减少 IO 次数,提高效率。

  printf(…) 在 glibc 中默认为行缓冲,遇到以下几种情况会刷新缓冲区,输出内容:

  (1)缓冲区填满;

  (2)写入的字符中有换行符或回车符;

  (3)调用 fflush(…) 手动刷新缓冲区;

  (4)调用 scanf(…) 从输入缓冲区中读取数据时,也会将输出缓冲区内的数据刷新。

  可使用关闭行缓冲,或者设置新的缓冲区,uBuff 为自己指定的缓冲区。也可以使用来改变标准输出为全缓冲。全缓冲与行缓冲的区别在于遇到换行符不刷新缓冲区。

  printf(…) 在 VC++ 中默认关闭缓冲区,输出时会及时输到屏幕 [ 3 ] ^{[3]} [3]。如果显示开启缓冲区,只能设置全缓冲。因为微软闭源,所以无法研究 printf(…) 的实现源码。

  Linux 和 Windows 下的缓冲区管理可见:C的全缓冲、行缓冲和无缓冲 http://blog.csdn.net/k346k346/article/details/63259524)。

  5.printf 与 wprintf 不能同时使用

  该小结写在 2018 年 1 月15 日。两年后的今日,在网上苦苦搜索寻求答案,终于解决了之前的疑惑。

  在输出宽字符串时,发现将 printf 和 wprintf 同时使用时,则后使用的函数没有输出。这里建议不要同时使用 printf 和 wprintf,以免发生错误。

  printf 和 wprintf 不能同时输出宽字符串的示例代码如下:

  上面的代码中语句 1 和语句 2 不能同时存在,否则只能正常输出第一个。也不知道在 Windows 平台是否也存在这种问题,有兴趣的读者可以尝试一下。关于原因,GNU 官方文档中有明确说明不能同时使用 printf 与 wprintf,参见The GNU C Library Section 12.6 Streams in Internationalized Applications http://www.gnu.org/software/libc/manual/html_node/Streams-and-I18N.html#Streams-and-I18N),内容如下:

  这里是因为输出流在被创建时,不存在流定向,一旦使用了 printf(多字节字符串) 或 wprintf(宽字符串)后,就被设置为对应的流定向,且无法更改。可以使用如下函数当前输出流的流定向。

  通过 fwide 可以设置当前流定向,前提是未有任何的 I/O 操作,也就是当前流尚未被设置任何流定向。顺带吐槽一下,不知为何标准库函数 fwide 实现的如此受限。具体操作如下:

  既然 GNU C 存在这个问题,那该如何解决呢?这里有两种办法:

  (1)统一使用一种函数。

  例如:

  (2)使用 C 标准库函数 freopen(…) 清空流定向。

  上面可以让 printf(…) 与 wprintf(…) 同时使用。

  6.小结

  耗时很久,终于完成了此篇看似基础,但却纷繁复杂的 printf(…) 用法。由于时间和个人水平有限,文章不足之处在所难免,也请读者批评指正,不甚感激。

2024最新激活全家桶教程,稳定运行到2099年,请移步至置顶文章:https://sigusoft.com/99576.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。 文章由激活谷谷主-小谷整理,转载请注明出处:https://sigusoft.com/94559.html

(0)
上一篇 2024年 5月 26日
下一篇 2024年 5月 26日

相关推荐

关注微信